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内 容 提 要 
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本 书 适合 SQL 初学 者 ， 也 可 供 广 大 开发 及 管理 人 员 人 参考 。 
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SQL 是 使 用 最 为 广泛 的 数据 库 语 言 。 不 管 你 是 应 用 开发 者 、 数 据 库 管 理 
员 、Web 应 用 设计 师 、 移 动 应 用 开发 人 员 ,， 还 是 只 使 用 Microsoft Office, 
掌握 良好 的 SQL 知识 对 用 好 数据 库 都 是 很 重要 的 。 



































本 书 可 以 说 是 应 需 而 生 。 我 讲授 了 多 年 的 Web 应 用 开发 ， 学 生 们 经 党 要 
求 我 推荐 一 些 SQL KIB. SQL 方面 的 书 很 多 ， 有 的 其 实 很 不 错 , 但 它们 
都 有 一 个 共同 的 特点 ， 就 是 讲授 的 内 容 太 多 了 ， 多 数 人 其 实 不 需要 了 解 
那么 多 。 很 多 图 书 讲 的 不 是 SQL 本 身 ， 而 是 从 数据 库 设计 、 规 范 化 到 关 
系数 据 库 理论 以 及 管理 问题 等 ， 事 无 巨细 都 讲 一 通 。 当 然 ， 这 些 内 容 也 
很 重要 ， 但 大 多 数 读者 仅 想 学 习 SQL， 他 们 未 必 感 兴趣 。 























因此 ,我 找 不 到 合适 书籍 推荐 给 学 生 , 只 好 把 在 课堂 上 给 学 生 讲 授 的 SQL 
知识 汇编 成 了 本 书 。 本 书 将 讲授 读者 需要 了 解 的 SQL 知识 ， 从 简单 的 数 
据 检索 入 手 ， 逐 步 过 渡 到 一 些 较为 复杂 的 内 容 ， 如 联结 、 子 查询 、 存 储 
过 程 、 游 标 、 触 发 器 以 及 表 约 束 等 。 读 者 将 从 本 书 中 循序 渐进 、 系 统 而 
直接 地 学 到 SQL 的 知识 和 技巧 。 
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本 书写 到 了 第 4 版 , 它 已 经 教会 了 英语 国家 近 30 万 的 读者 使 用 SQL, 并 





1 课 ， 











且 还 翻译 出 版 了 十 多 种 其 他 语言 的 版 本 。 现 在 轮 到 你 了 ， 让 我 们 翻 到 第 


开始 学 习 吧 。 你 将 很 快 编写 出 世界 级 的 SQL。 


读者 对 象 


AP 


适合 以 下 读者 : 

a SQL 新 手 ; 

口 希望 快速 学 会 并 熟练 使 用 SQL; 

口 希望 知道 如 何 使 用 SQL 开发 应 用 程序 ; 

口 希望 在 无 人 帮助 的 情况 下 有 效 而 快速 地 使 用 SQL。 





本 书 涵盖 的 DBMS 


一 般 来 说 ， 本 书 中 所 讲授 的 SQL 可 以 应 用 到 任何 数据 库 管理 系统 
(DBMS ), 但 是 , 各 种 SQL 实现 不 尽 相 同 ， 本 书 介绍 的 SQL 主要 适用 于 


以 下 



































系统 〈 需 要 时 会 给 出 特殊 说 明和 注释 ): 

D Apache Open Office Base; 

J IBM DB2; 

DQ Microsoft Access; 

O Microsoft SQL Server (包括 Microsoft SQL Server Express ); 
口 MariaDB ; 

口 MySQL; 

口 Oracle( 包括 Oracle Express ); 

口 PostgreSQL ; 

DQ SQLite。 
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本 书 中 的 所 有 数据 库 示例 ( 或 者 创建 数据 库 示 例 的 SQL 脚本 例子 ) 对 于 
这 些 DBMS 都 是 适用 的 ， 它 们 可 以 在 本 书 的 网 页 http://forta.com/books/ 
0672336073/ 上 获得 。 





本 书 约定 


本 书 采用 等 宽 字 体 表示 代码 ， 读 者 输入 的 文本 与 应 该 出 现在 屏幕 上 的 文 


等 宽 字 
本 也 都 以 等 宽 字体 给 出 。 








It will look like this to mimic the way text looks on your screen. 








变量 和 表达 式 的 占 位 符 用 斜体 表示 ， 你 可 以 用 具体 的 值 代替 它 。 





代码 行 前 的 箭头 (ee ) 表示 代码 太 长 ， 上 一 行 容纳 不 下 。 在 加 符号 后 输 
入 的 所 有 字符 都 应 该 是 前 一 行 的 内 容 。 








说 明 
给 出 上 下 文 讨论 中 比较 重要 的 信息 








提示 
就 某 任 务 给 出 建议 或 更 简单 的 方法 。 








注意 
提醒 可 能 出 现 的 问题 ， 避 免 出 现 事 故 。 








新 术语 
清晰 定义 重要 的 新 词汇 。 
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pip 


输入 了 
读者 可 以 自己 输入 的 代码 ， 通 常 紧 挨 着 代码 出 现 。 
输出 了 
强调 某 个 程序 执行 时 的 输出 ， 通 常 出 现在 代码 后 。 
BY 


对 程序 代码 进行 逐 行 分 析 。 
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第 1 课 了 解 SQL 








1.1 数据 库 基 础 

你 正在 读 这 本 SQL 图 书 , 这 表明 你 需要 以 某 种 方式 与 数据 库 打 交道 ,SQL 
正 是 用 来 实现 这 一 任务 的 语言 ， 因 此 在 学 习 SQL 之 前 ,你 应 该 对 数据 库 
及 数据 库 技术 的 某 些 基本 概念 有 所 了 解 。 




















你 可 能 还 没有 意识 到 ， 其 实 自己 一 直 在 使 用 数据 库 。 每 当 你 从 电子 邮件 
地 址 夭 里 查找 名 字 时 ， 就 是 在 使 用 数据 库 。 你 在 网 站 上 进行 搜索 ， 也 是 
在 使 用 数据 库 。 你 在 工作 中 登录 网 络 ， 也 需要 依靠 数据 库 验 证 用 户 名 和 
密码 。 即 使 是 在 自动 取款 机 上 使 用 ATM 卡 , 也 要 利用 数据 库 进 行 密码 验 
证 和 查询 余额 。 



























































虽然 我 们 一 直 都 在 使 用 数据 库 , 但 对 究竟 什么 是 数据 库 并 不 十 分 清楚 。 更 
何况 人 们 可 能 会 使 用 相同 的 数据 库 术 语 表 示 不 同 的 事物 ,进一步 加 剧 了 这 
种 混乱 。 因 此 ， 我 们 首先 给 出 一 些 最 重要 的 数据 库 术 语 ， 并 加 以 说 明 。 























提示 : 基本 概念 回顾 
后 面 是 一 些 基 本 数据 库 概 念 的 简要 介绍 。 如 果 你 已 经 具有 一 定 的 数 
据 库 经 验 ， 可 以 借 此 复习 巩固 一 下 ; 如 果 你 刚 开 始 接触 数据 库 ， 可 
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以 由 此 了 解 必 需 的 基本 知识 。 理 解数 据 库 概念 是 掌握 SQL 的 重要 前 
提 ， 如 果 有 必要 ， 你 或 许 还 应 该 参阅 其 他 一 些 有 关 数 据 库 基础 知识 
的 书籍 。 





1.1.1 数据 库 


数据 库 这 个 术语 的 用 法 很 多 ， 但 就 本 书 而 言 ( 从 SQL 的 角度 来 看 )， 数 
据 库 是 一 个 以 某 种 有 组 织 的 方式 存储 的 数据 集合 。 最 简单 的 办 法 是 将 数 
据 库 想 象 为 一 个 文件 柜 。 这 个 文件 柜 是 一 个 存放 数据 的 物理 位 置 ， 不 管 
数据 是 什么 ， 也 不 管 数据 是 如 何 组 织 的 。 
























































数据 库 ( database ) 
保存 有 组 织 的 数据 的 容器 (通常 是 一 个 文件 或 一 组 文件 )。 








注意 : 误 用 导致 混淆 

人 们 通常 用 数据 库 这 个 术语 来 代表 他 们 使 用 的 数据 库 软 件 ， 这 是 不 正 
确 的 ， 也 因此 产生 了 许多 混淆 。 确 切 地 说 ， 数 据 库 软件 应 称 为 数据 库 
管理 系统 (DBMS )。 数 据 库 是 通过 DBMS 创建 和 操纵 的 容器 ， 而 具 
体 它 究竟 是 什么 ， 形 式 如 何 ， 各 种 数据 库 都 不 一 样 。 











11.2 R 


你 往 文件 柜 里 放 资 料 时 ， 并 不 是 随便 将 它们 扔 进 某 个 抽 居 就 完事 了 ， 而 
是 在 文件 柜 中 创建 文件 ， 然 后 将 相关 的 资料 放 入 特定 的 文件 中 。 




















在 数据 库 领 域 中 ， 这 种 文件 称 为 表 。 表 是 一 种 结构 化 的 文件 ， 可 用 来 存 
储 某 种 特定 类 型 的 数据 。 表 可 以 保存 顾客 清单 、 产 品目 录 ， 或 者 其 他 信 
息 清单 ° 
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表 ( table ) 
某 种 特定 类 型 数据 的 结构 化 清单 。 


























I 





这 里 的 关键 一 点 在 于 ， 存 储 在 表 中 的 数据 是 同一 种 类 型 的 数据 或 清单 。 
决 不 应 该 将 顾客 的 清单 与 订单 的 清单 存储 在 同一 个 数据 库 表 中 ， 否 则 以 
后 的 检索 和 访问 会 很 困难 。 应 该 创建 两 个 表 ， 每 个 清单 一 个 表 。 

















数据 库 中 的 每 个 表 都 有 一 个 名 字 来 标识 自己 。 这 个 名 字 是 唯一 的 ， 即 数 
据 库 中 没有 其 他 表 具 有 相同 的 名 字 。 














说 明 : RA 

使 表 名 成 为 唯一 的 ， 实 际 上 是 数据 库 名 和 表 名 等 的 组 合 。 有 的 数据 库 
还 使 用 数据 库 拥 有 者 的 名 字 作 为 唯一 名 的 一 部 分 。 也 就 是 说 ， 虽 然 在 
相同 数据 库 中 不 能 两 次 使 用 相同 的 表 名 ， 但 在 不 同 的 数据 库 中 完全 可 
以 使 用 相同 的 表 名 。 














表 具 有 一 些 特性 ， 这 些 特性 定义 了 数据 在 表 中 如 何 存储 ， 包 含 存储 什么 
样 的 数据 ， 数 据 如 何 分 解 ， 各 部 分 信息 如 何 命名 等 信息 。 描 述 表 的 这 组 
信息 就 是 所 谓 的 模式 〈schema )， 模 式 可 以 用 来 描述 数据 库 中 特定 的 表 ， 
也 可 以 用 来 描述 整个 数据 库 ( 和 其 中 表 的 关系 )。 




















模式 








1.1.3” 列 和 数据 类 型 
表 由 列 组 成 。 列 存储 表 中 某 部 分 的 信息 。 








列 ( column ) 


表 中 的 一 个 字段 。 所 有 表 都 是 由 一 个 或 多 个 列 组 成 的 。 
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理解 列 的 最 好 办 法 是 将 数据 库 表 想象 为 一 个 网 格 , 就 像 个 电子 表格 那样 。 
网 格 中 每 一 列 存储 着 某 种 特定 的 信息 。 例 如 ， 在 顾客 表 中 ， 一 列 存储 顾 
客 编号 ， 另 一 列 存储 顾客 姓名 ， 而 地 址 、 城 市 、 州 以 及 邮政 编码 全 都 存 
储 在 各 自 的 列 中 。 











提示 : 分 解数 据 

正确 地 将 数据 分 解 为 多 个 列 极为 重要 。 例 如 ， 城 市 、 州 、 邮 政 编码 
应 该 总 是 彼此 独立 的 列 。 通 过 分 解 这 些 数据 ， 才 有 可 能 利用 特定 的 
列 对 数据 进行 分 类 和 过 滤 〈 如 找 出 特定 州 或 特定 城市 的 所 有 顾客 )。 
如 果 城 市 和 州 组 合 在 一 个 列 中 ， 则 按 州 进行 分 类 或 过 滤 就 会 很 
困难 。 

你 可 以 根据 自己 的 具体 需求 来 决定 把 数据 分 解 到 何 种 程度 。 例 如 ， 一 


般 可 以 把 门牌 号 和 街道 名 一 起 存储 在 地 址 里 。 这 没有 问题 ， 除 非 你 哪 
天 想 用 街道 名 来 排序 ， 这 时 ， 最 好 将 门牌 号 和 街道 名 分 开 。 











数据 库 中 每 个 列 都 有 相应 的 数据 类 型 。 数 据 类 型 (datatype ) 定义 了 列 可 
以 存储 哪些 数据 种 类 。 例 如 ， 如 果 列 中 存储 的 是 数字 (或 许 是 订单 中 的 

品 数 )， 则 相应 的 数据 类 型 应 该 为 数值 类 型 。 如 果 列 中 存储 的 是 日 期 、 
文本 、 注 释 、 金 额 等 ， 则 应 该 规定 好 恰当 的 数据 类 型 。 


















































数据 类 型 
所 允许 的 数据 的 类 型 。 每 个 表 列 都 有 相应 的 数据 类 型 ， 它 限制 (或 允 
许 ) 该 列 中 存储 的 数据 。 























数据 类 型 限定 了 可 存储 在 列 中 的 数据 种 类 ( 例如 ， 防 止 在 数值 字段 中 录 
入 字符 值 ),。 数据 类 型 还 帮助 正确 地 分 类 数据 , 并 在 优化 磁盘 使 用 方面 起 
重要 的 作用 。 因 此 ， 在 创建 表 时 必须 特别 关注 所 用 的 数据 类 型 。 
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注意 : 数据 类 型 兼容 

数据 类 型 及 其 名 称 是 SQL 不 兼容 的 一 个 主要 原因 。 虽 然 大 多 数 基 本 数 
据 类 型 得 到 了 一 致 的 支持 , 但 许多 高 级 的 数据 类 型 却 没有 。 BRE, 
偶然 会 有 相同 的 数据 类 型 在 不 同 的 DBMS 中 具有 不 同 的 名 称 。 对 此 用 
户 毫 无 办 法 ， 重 要 的 是 在 创建 表 结 构 时 要 记 住 这 些 差异 。 








1.1.4 íT 


表 中 的 数据 是 按 行 存储 的 ， 所 保存 的 每 个 记录 存储 在 自己 的 行内 。 如 果 
将 表 想 象 为 网 格 ， 网 格 中 垂直 的 列 为 表 列 ， 水 平行 为 表 行 。 





例如 ,顾客 表 可 以 每 行 存储 一 个 顾客 。 表 中 的 行 编号 为 记录 的 编号 。 





fT (row ) 
表 中 的 一 个 记录 。 








说 明 : 是 记录 还 是 行 ? 
你 可 能 听 到 用 户 在 提 到 行 时 称 其 为 数据 库 记 录 (record )。 这 两 个 术语 
多 半 是 可 以 交替 使 用 的 ， 但 从 技术 上 说 ， 行 才 是 正确 的 术语 。 











1.1.5 主键 
表 中 每 一 行 都 应 该 有 一 列 〈 或 几 列 ) 可 以 唯一 标识 自己 。 顾 客 表 可 以 使 
用 顾客 编号 ， 而 订单 表 可 以 使 用 订单 ID 。 雇 员 表 可 以 使 用 雇员 ID 或 雇 


员 社 会 安全 号 。 


























主键 ( primary key ) 
一 列 (或 一 组 列 )， 其 值 能 够 唯一 标识 表 中 每 一 行 。 
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唯一 标识 表 中 每 行 的 这 个 列 ( 或 这 儿 列 ) 称 为 主键 。 主 键 用 来 表示 一 个 
特定 的 行 。 没 有 主键 ,更 新 或 删除 表 中 特定 行 就 极为 困难 ， 因 为 你 不 能 
保证 操作 只 涉及 相关 的 行 。 








提示 : 应 该 总 是 定义 主键 
虽然 并 不 总 是 需要 主键 ,但 多 数 数据 库 设计 者 都 会 保证 他 们 创建 的 每 
个 表 有 具有 一 个 主键 ， 以 便于 以 后 的 数据 操作 和 管理 。 














表 中 的 任何 列 都 可 以 作为 主键 ， 只 要 它 满足 以 下 条 件 : 


D 任意 两 行 都 不 具有 相同 的 主键 值 ; 

D 每 一 行 都 必须 具有 一 个 主键 值 ( 主键 列 不 允许 NULL 值 ); 

口 主键 列 中 的 值 不 允许 修改 或 更 新 ; 

口 主键 值 不 能 重用 ( 如 果 某 行 从 表 中 删除 , 它 的 主键 不 能 赋 给 以 后 的 新 行 )。 




















主键 通常 定义 在 表 的 一 列 上 ， 但 并 不 是 必需 这 么 做 ， 也 可 以 一 起 使 用 多 
个 列 作为 主键 。 在 使 用 多 列 作为 主键 时 ， 上 述 条 件 必 须 应 用 到 所 有 列 ， 
所 有 列 值 的 组 合 必须 是 唯一 的 〈 但 单个 列 的 值 可 以 不 唯一 ) 




















还 有 一 种 非常 重要 的 键 ， 称 为 外 键 ， 我 们 将 在 第 12 课 中 介绍 。 


1.2 什么 是 SQL 


SQL ( 发 音 为 字母 S-Q-L zk sequel ) 是 Structured Query Language ( 结构 
化 查询 语言 ) 的 缩写 。SQL 是 一 种 专门 用 来 与 数据 库 沟 通 的 语言 。 


与 其 他 语言 ( 如 英语 或 Java. C. PHP 这 样 的 编程 语言 ) 不 一 样 ，SQL 
中 只 有 很 少 的 词 ， 这 是 有 意 而 为 的 。 设 计 SQL 的 目的 是 很 好 地 完成 一 项 
任务 一 一 提供 一 种 从 数据 库 中 读 写 数据 的 简单 有 效 的 方法 。 
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SQL 有 如 下 的 优点 。 








D SQL 不 是 某 个 特定 数据 库 供应 商 专 有 的 语言 。 几 乎 所 有 重要 的 DBMS 
都 支持 SQL， 所 以 学 习 此 语言 使 你 几乎 能 与 所 有 数据 库 打交道 。 

口 SQL 简单 易学 。 它 的 语句 全 都 是 由 有 很 强 描述 性 的 英语 单词 组 成 ， 而 
且 这 些 单词 的 数目 不 多 。 
口 SQL 虽然 看 上 去 很 简单 ， 但 实际 上 是 一 种 强 有 力 的 语言 ， 灵 活 使 用 其 
语言 元 素 ， 可 以 进行 非常 复杂 和 高 级 的 数据 库 操作 。 












































下 面 我 们 将 开始 真正 学 习 SQL。 





说 明 : SQL 的 扩展 

许多 DBMS 厂商 通过 增加 语句 或 指令 ,对 SQL 进行 了 扩展 。 这 种 扩 
展 的 目的 是 提供 执行 特定 操作 的 额外 功能 或 简化 方法 。 虽 然 这 种 扩展 
RAA, 但 一 般 都 是 针对 个 别 DBMS 的 , 很 少 有 两 个 以 上 的 供应 商 支 
持 这 种 扩展 。 

标准 SQL 由 ANSI 标准 委员 会 管理 ， 从 而 称 为 ANSI SQL。 所 有 主要 


的 DBMS， 即 使 有 自己 的 扩展 ， 也 都 支持 ANSI SQL。 各 个 实现 有 自 
己 的 名 称 ， 如 PL/SQL, Transact-SQL 等 。 


本 书 讲 授 的 SQL 主要 是 ANSI SQL 。 在 使 用 某 种 DBMS 特定 的 SQL 
时 ， 会 特别 说 明 。 











13 ”动手 实践 


与 其 他 任何 语言 一 样 ， 学 习 SQL 的 最 好 方法 是 自己 动手 实践 。 为 此 , 需 
要 一 个 数据 库 和 用 来 测试 SQL 语句 的 应 用 系统 。 
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本 书 中 所 有 课程 采用 的 都 是 真实 的 SQL 语句 和 数据 表 。 附录 A 给 出 了 具 
体 的 样 例 表 ， 并 介绍 了 获得 (或 创建 ) 它们 的 详细 步骤 ， 便 于 读者 理解 
每 一 课 讲授 的 内 容 。 附 录 B 介 绍 在 各 种 应 用 程序 中 执行 SQL 所 需 的 步骤 。 
在 进入 下 一 课 之 前 ， 强 烈 建议 读者 先 阅读 这 两 个 附录 的 内 容 ， 为 以 后 的 
学 习 做 好 准备 。 











14 “小结 


这 一 课 介绍 了 什么 是 SQL, 它 为 什么 很 有 用 。 因 为 SQL 是 用 来 与 数据 库 
打交道 的 ， 所 以 ,我 们 也 复习 了 一 些 基 本 的 数据 库 术 语 。 
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第 2 课 检索 数据 


这 一 课 介绍 如 何 使 用 SELECT 语句 从 表 中 检索 一 个 或 多 个 数据 列 。 





2.1 SELECT 语句 








正如 第 1 课 所 述 ，SQL 语句 是 由 简单 的 英语 单词 构成 的 。 这 些 单词 称 为 
KEF, 每 个 SQL 语句 都 是 由 一 个 或 多 个 关键 字 构 成 的 。 最 经 常 使 用 的 
SQL 语句 大 概 就 是 SELECT 语句 了 。 它 的 用 途 是 从 一 个 或 多 个 表 中 检索 
信息 


Huso 

















关键 字 ( keyword ) 
作为 SQL 组 成 部 分 的 保留 字 。 关 键 字 不 能 用 作 表 或 列 的 名 字 。 附 录 下 
列 出 了 某 些 经 常 使 用 的 保留 字 。 











为 了 使 用 SELECT 检索 表 数 据 ， 必 须 至 少 给 出 两 条 信息 一 一 想 选 择 什么 ， 
以 及 从 什么 地 方 选择 。 








说 明 : 理解 例子 

本 书 各 课程 中 的 样 例 SQL 语句 (和 样 例 输出 ) 使 用 了 附录 A 中 描述 的 
一 组 数据 文件 。 如 果 想 要 理解 和 试验 这 些 样 例 (我 强烈 建议 这 样 做 )， 
请 参阅 附录 A， 它 解释 了 如 何 下 载 或 创建 这 些 数据 文件 。 
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重要 的 是 ， 要 理解 SQL 是 一 种 语言 而 不 是 一 个 应 用 程序 。 具 体 如 何 写 
SQL 语 名 并 显示 语句 输出 ， 是 随 不 同 的 应 用 程序 而 变化 的 。 为 帮助 读 
者 根据 自己 的 环境 使 用 相应 的 例子 ， 附 录 B 介绍 了 如 何 针 对 许多 流行 
的 应 用 程序 及 开发 环境 发 出 本 书 中 介绍 的 语句 。 如 果 读 者 需要 了 解 菜 
个 应 用 程序 ， 附 录 B 中 也 给 出 了 相应 的 建议 。 








22 ”检索 单个 列 
我 们 将 从 简单 的 SQL SELECT 语句 讲 起 ， 此 语句 如 下 所 示 : 


输入 了 


SELECT prod_name 
FROM Products; 


SM Y 

上 述 语句 利用 SELECT 语句 从 Products 表 中 检索 一 个 名 为 prod_name 
的 列 。 所 需 的 列 名 写 在 SELECT 关键 字 之 后 ，FROM 关键 字 指 出 从 哪个 表 
中 检索 数据 。 此 语句 的 输出 如 下 所 示 : 








输出 


prod_name 

Fish bean bag toy 
Bird bean bag toy 
Rabbit bean bag toy 
8 inch teddy bear 
12 inch teddy bear 
18 inch teddy bear 
Raggedy Ann 

King doll 

Queen doll 
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说 明 : 未 排序 数据 

如 果 你 自己 试验 这 个 查询 ， 可 能 会 发 现 显 示 输 出 的 数据 顺序 与 这 里 的 
不 同 。 出 现 这 种 情况 很 正常 。 如 果 没 有 明确 排序 查询 结果 (下 一 课 介 
绍 ), 则 返回 的 数据 没有 特定 的 顺序 。 返 回 数据 的 顺序 可 能 是 数据 被 添 
加 到 表 中 的 顺序 , 也 可 能 不 是 。 只 要 返回 相同 数目 的 行 , 就 是 正常 的 。 














如 上 的 一 条 简单 SELECT 请 句 将 返回 表 中 的 所 有 行 。 数 据 没 有 过 滤 (过 
滤 将 得 出 结果 集 的 一 个 子 集 )， 也 没有 排序 。 以 后 几 课 将 讨论 这 些 内 容 。 











提示 : 结束 SQL 语句 
多 条 SQL 语句 必须 以 分 号 (; ) 分 隔 。 多 数 DBMS 不 需要 在 单条 SQL 
语句 后 加 分 号 ， 但 也 有 DBMS 可 能 必须 在 单条 SQL 语句 后 加 上 分 号 。 
当然 ， 如 果 愿 意 可 以 总 是 加 上 分 号 。 事 实 上 ， 即 使 不 一 定 需 要 ， 加 上 
分 号 也 肯定 没有 坏处 。 








提示 : SQL 语句 和 大 小 写 

请 注意 ，SQL 语句 不 区 分 大 小 写 ， 因 此 SELECT 与 select 是 相同 的 。 
同样 ,写成 Select 也 没有 关系 。 许多 SQL 开发 人 员 喜 欢 对 SQL 关键 
字 使 用 大 写 ， 而 对 列 名 和 表 名 使 用 小 写 ， 这 样 做 使 代码 更 易于 阅读 和 
调试 。 不 过 ， 一 定 要 认识 到 虽然 SQL 是 不 区 分 大 小 写 的 ， 但 是 表 名 、 
列 名 和 值 可 能 有 所 不 同 ( 这 有 赖 于 具体 的 DBMS 及 其 如 何 配置 )。 








提示 : 使 用 空格 
在 处 理 SQL 语句 时 ,其 中 所 有 空格 都 被 忽略 。SQL 语句 可 以 写成 长 长 
的 一 行 ， 也 可 以 分 写 在 多 行 。 下 面 这 3 种 写法 的 作用 是 一 样 的 。 


SELECT prod_name 
FROM Products; 


SELECT prod_name FROM Products; 
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SELECT 
prod_name 
FROM 
Products ; 





多 数 SQL 开发 人 员 认 为 ， 将 SQL 语句 分 成 多 行 更 容易 阅读 和 调试 。 





23 ”检索 多 个 列 


要 想 从 一 个 表 中 检索 多 个 列 ， 仍 然 使 用 相同 的 SELECT 语句 。 唯 一 的 不 
同 是 必须 在 SELECT 关键 字 后 给 出 多 个 列 名 ， 列 名 之 间 必 须 以 逗号 分 隔 。 














提示 : 当心 逗号 
在 选择 多 个 列 时 ， 一定 要 在 列 名 之 闻 加 上 如 号 ， 但 最 后 一 个 列 名 后 不 
加 。 如 果 在 最 后 一 个 列 名 后 加 了 运 号 ， 将 出 现 错误 。 











下 面 的 SELECT 语句 从 Products 表 中 选择 3 列 。 





输入 了 


SELECT prod_id，prod_name，prod_price 
FROM Products; 


BY 


与 前 一 个 例子 一 样 , 这 条 语句 使 用 SELECT 语句 从 表 Products 中 选择 数 
据 。 在 这 个 例子 中 ， 指 定 了 3 个 列 名 ， 列 名 之 间 用 逗号 分 隔 。 此 语句 的 
输出 如 下 : 











输出 


prod_id prod_name prod_price 
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BNBGO1 Fish bean bag toy 3.4900 
BNBG02 Bird bean bag toy 3.4900 
BNBGO3 Rabbit bean bag toy 3.4900 
BRO1 8 inch teddy bear 5.9900 
BR02 12 inch teddy bear 8.9900 
BRO3 18 inch teddy bear 11.9900 
RGAN01 Raggedy Ann 4.9900 
RYLO1 King doll 9.4900 
RYLO2 Queen dool 9.4900 





说 明 : 数据 表示 

从 上 述 输出 可 以 看 到 ，SQL 语句 一 般 返 回 原始 的 、 无 格式 的 数据 。 数 
据 的 格式 化 是 表示 问题 ， 而 不 是 检索 问题 。 因 此 ,表示 ( 如 把 上 面 的 
价格 值 显示 为 正确 的 十 进 制 数值 货币 金额 ) 一 般 在 显示 该 数据 的 应 用 
程序 中 规定 。 通 常 很 少 直接 使 用 实际 检索 出 的 数据 (没有 应 用 程序 提 
供 的 格式 )。 











24 检索 所 有 列 

除了 指定 所 需 的 列 外 ( 如 上 所 述 ， 一 个 或 多 个 列 )，SELECT 语句 还 可 以 
检索 所 有 的 列 而 不 必 逐 个 列 出 它们 。 在 实际 列 名 的 位 置 使 用 星 号 (* ) 通 
配 符 可 以 做 到 这 点 ， 如 下 所 示 。 

输入 可 


SELECT * 
FROM Products; 





BY 


如 果 给 定 一 个 通配符 〈* )， 则 返回 表 中 所 有 列 。 列 的 顺序 一 般 是 列 在 表 
定义 中 出 现 的 物理 顺序 , 但 并 不 总 是 如 此 。 不 过 , SQL 数据 很 少 这样 ( 通 
常 ， 数 据 返 回 给 应 用 程序 ， 根 据 需要 进行 格式 化 ， 再 表示 出 来 ) 因此， 
这 不 应 该 造成 什么 问题 。 
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注意 : 使 用 通配符 

一 般 而 言 ， 除 非 你 确实 需要 表 中 的 每 一 列 ， 否 则 最 好 别 使 用 * 通 配 符 。 
虽然 使 用 通配符 能 让 你 自己 省 事 ， 不 用 明确 列 出 所 需 列 ， 但 检索 不 需 
要 的 列 通 常会 降低 检索 和 应 用 程序 的 性 能 。 








提示 : 检索 未 知 列 
使 用 通配符 有 一 个 大 优点 。 由 于 不 明确 指定 列 名 (因为 星 号 检索 每 一 
列 )， 所 以 能 检索 出 名 字 未 知 的 列 。 








25 ”检索 不 同 的 值 


如 前 所 述 ，SELECT 语句 返回 所 有 匹配 的 行 。 但是， 如 果 你 不 希望 每 个 值 
每 次 都 出 现 ， 该 怎么 办 呢 ?” 例 如 ， 你 想 检 索 products 表 中 所 有 产品 供 
应 商 的 ID: 











输入 了 


SELECT vend_id 
FROM Products; 


输出 


vend_id 
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SELECT 语句 返回 9 行 ( 即 使 表 中 只 有 3 个 产品 供应 商 ), 因为 products 
RPA 9 种 产品 。 那 么 如 何 检索 出 不 同 的 值 ? 











办 法 就 是 使 用 DISTINCT 关键 字 ， 顾 名 思 义 ， 它 指示 数据 库 只 返回 不 同 
的 值 。 


输入 了 


SELECT DISTINCT vend_id 
FROM Products; 


BY 





SELECT DISTINCT vend_id 告诉 DBMS 只 返回 不 同 (具有 唯一 性 ) 的 
vend_id 行 ， 所 以 正如 下 面 的 输出 ， 只 有 3 行 。 如 果 使 用 DISTINCT 关 
键 字 ， 它 必须 直接 放 在 列 名 的 前 面 。 





























注意 : 不 能 部 分 使 用 DISTINCT 

DISTINCT 关键 字 作用 于 所 有 的 列 ， 不 仅仅 是 跟 在 其 后 的 那 一 列 。 例 
如 ， 你 指定 SELECT DISTINCT vend_id，prod_price， 除 非 指 定 的 
两 列 完全 相同 ， 否 则 所 有 的 行 都 会 被 检索 出 来 。 











26 ”限制 结果 


SELECT 语句 返回 指定 表 中 所 有 匹配 的 行 , 很 可 能 是 每 一 行 。 如 果 你 只 术 























[m 
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返回 第 一 行 或 者 一 定数 量 的 行 ， 该 怎么 办 呢 ? 这 是 可 行 的 ， 然 而 遗憾 的 
是 ， 各 种 数据 库 中 的 这 一 SQL 实现 并 不 相同 。 











在 SQL Server 和 Access 中 使 用 SELECT 时 , 可 以 使 用 TOP 关键 字 来 限制 
最 多 返回 多 少 行 ， 如 下 所 示 : 











输入 了 


SELECT TOP 5 prod_name 
FROM Products; 


输出 了 


prod_name 

8 inch teddy bear 
12 inch teddy bear 
18 inch teddy bear 
Fish bean bag toy 
Bird bean bag toy 


Sir Y 
上 面 代码 使 用 SELECT TOP 5 语句 ， 只 检索 前 $ 行 数据 。 





如 果 你 使 用 的 是 DB2， 很 可 能 习惯 使 用 下 面 这 一 DBMS 特定 的 SQL H 
句 ， 像 这 样 : 

输入 了 

SELECT prod_name 


FROM Products 
FETCH FIRST 5 ROWS ONLY; 


BY 








FETCH FIRST 5 ROWS ONLY 就 会 按 字面 的 意思 去 做 的 。 
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如 果 你 使 用 Oracle， 需 要 基于 ROWNUM ( 行 计数 器 ) 来 计算 行 ， 像 这 样 : 


输入 了 


SELECT prod_name 
FROM Products 
WHERE ROWNUM <=5; 





如 果 你 使 用 MySQL .MariaDB .PostgreSQL 或 者 SQLite, 需要 使 用 LIMIT 
子 句 ， 像 这 样 : 

输入 到 

SELECT prod_name 

FROM Products 

LIMIT 5; 


BY 


上 述 代码 使 用 SELECT 语句 来 检索 单独 的 一 列 数 据 。LIMIT 5 指示 MySQL 
等 DBMS 返回 不 超过 5 行 的 数据 。 这 个 语句 的 输出 参见 下 面 的 代码 。 





为 了 得 到 后 面 的 5 行 数据 , 需要 指定 从 哪儿 开始 以 及 检索 的 行 数 , 像 这 样 : 





输入 了 


SELECT prod_name 
FROM Products 
LIMIT 5 OFFSET 5; 


BY 


LIMIT 5 OFFSET 5 指示 MySQL 等 DBMS 返回 从 第 5 行 起 的 5 行 数据 。 
第 一 个 数字 是 指 从 哪儿 开始 ， 第 二 个 数字 是 检索 的 行 数 。 这 个 语句 的 输 


出 是 : 
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输出 


prod_name 


Rabbit bean bag toy 
Raggedy Ann 

King doll 

Queen doll 


所 以 ，LIMIT 指定 返回 的 行 数 。LIMIT 带 的 OFFSET 指定 从 哪儿 开始 。 
在 我 们 的 例子 中 ，Products 表 中 只 有 9 种 产品 ， 所 以 LIMIT 5 OFFSET 
5 只 返回 了 4 行 数 据 ( 因为 没有 第 5 行 )。 











注意 : 第 0 行 
第 一 个 被 检索 的 行 是 第 0 行 ， 而 不 是 第 1 行 。 因 此 ，LIMIT 1 OFFSET 
工会 检索 第 2 行 ， 而 不 是 第 1 行 。 








提示 : MySQL 和 MariaDB 捷径 
MySQL 和 MariaDB 支持 简化 版 的 LIMIT 4 OFFSET 3346) , PP LIMIT 3,4。 
使 用 这 个 语法 ， 运 号 之 前 的 值 对 应 OFFSET， 运 号 之 后 的 值 对 应 LIMIT. 








说 明 : 并 非 所 有 的 SOL 实现 都 一 样 

我 加 入 这 一 节 只 有 一 个 原因 ， 就 是 要 说 明 ，SQL 虽然 通常 都 有 相当 一 
致 的 实现 ， 但 你 不 能 想当然 地 认为 它 总 是 这 样 。 非 常 基本 的 语句 往往 
是 容易 移植 的 ,但 较 复 杂 的 语句 就 不 同 了 。, 当 你 针对 菜 个 问题 寻找 SQL 
解决 方案 时 ， 一定 要 记 住 这 一 点 。 











27 ”使 用 注释 
可 以 看 到 ，SQL 语句 是 由 DBMS 处 理 的 指令 。 如 果 你 希望 包括 不 进行 处 理 
和 执行 的 文本 ， 该 怎么 办 呢 ? 为 什么 你 想 要 这 么 做 呢 ? 原因 有 以 下 几 点 。 
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O 我 们 这 里 使 用 的 SQL 语句 都 很 得， 也 很 简单 。 然 而 ， 随 着 SQL 语句 
变 长 ， 复 杂 性 增加 ， 你 就 会 想 添 加 一 些 描述 性 的 注释 ， 这 便于 你 自己 
今后 参考 ， 或 者 供 项 目 后 续 参 与 人 员 人 参考 。 这 些 注释 需要 骨 入 在 SQL 
脚本 中 ， 但 显然 不 能 进行 实际 的 DBMS 处 理 。( 相关 示例 可 以 参见 附 








录 B 中 使 用 的 create.sql 和 populate.sql。 ) 





口 这 同样 适用 于 SQL 文件 开始 处 的 内 容 ， 它 可 能 包含 程序 员 的 联系 方 
式 、 程 序 描述 以 及 一 些 说 明 。( 相关 示例 也 可 参见 附录 B 中 的 那些 .sql 


文件 。) 








口 注释 的 另 一 个 重要 应 用 是 暂时 停止 要 执行 的 SQL 代码 。 如 果 你 碰 到 一 
个 长 SQL 语句 ， 而 只 想 测 试 它 的 一 部 分 ， 那 么 应 该 注释 掉 一 些 代 码 ， 











以 便 DBMS 将 其 视 为 注释 而 加 以 忽略 。 
很 多 DBMS 都 支持 各 种 形式 的 注释 语法 。 我 们 先 来 看 行内 注释 : 


输入 了 


SELECT prod_name -- 这 是 一 条 注释 
FROM Products; 


BY 


注释 使 用 -- 〈 两 个 连 字 符 ) REITAN. -- 之 后 的 文本 就 是 注释 ， 例 如 ， 





这 用 来 描述 CREATE TABLE 语句 中 的 列 就 很 不 错 。 

















下 面 是 另 一 种 形式 的 行内 注释 ( 虽然 这 种 形式 很 少 得 到 支持 )。 








输入 了 

# 这 是 一 条 注释 
SELECT prod_name 
FROM Products; 
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BY 


在 一 行 的 开始 处 使 用 #， 这 一 整 行 都 将 作为 注释 。 你 在 本 书 提供 的 脚本 
create.sql 和 populate.sql 中 可 以 看 到 这 种 形式 的 注释 。 








你 也 可 以 进行 多 行 注 释 ， 注 释 可 以 在 脚本 的 任何 位 置 停止 和 开始 。 








输入 了 


/* SELECT prod_name, vend_id 
FROM Products; */ 

SELECT prod_name 

FROM Products; 


BY 

注释 从 /* 开 始 ， 到 */ 结 束 ，/* 和 */ 之 间 的 任何 内 容 都 是 注释 。 这 种 方式 
常用 于 给 代码 加 注释 ， 就 如 这 个 例子 演示 的 ， 这 里 定义 了 两 个 SELECT 
语句 ， 但 是 第 一 个 不 会 执行 ， 因 为 它 已 经 被 注释 掉 了 。 


























2.8 小 结 


这 一 课 学 习 了 如 何 使 用 SQL 的 SELECT 语句 来 检索 单个 表 列 、 多 个 表 列 
以 及 所 有 表 列 。 你 也 学 习 了 如 何 返回 不 同 的 值 ， 如 何 注释 代码 。 同 时 不 
幸 的 是 , 更 复杂 的 SQL 使 得 SQL 代码 变 得 不 轻便 。 下 一 课 将 讲授 如 何 对 
今 索 出 来 的 数据 进行 排序 。 
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这 一 课 讲授 如 何 使 用 SELECT 语句 的 ORDER BY 子 句 , 根据 需要 排序 检索 
出 的 数据 。 


3.1 ”排序 数据 


正如 上 一 课 所 述 ， 下面 的 SQL 语句 返回 某 个 数据 库 表 的 单个 列 。 但 请 看 
其 输出 ， 并 没有 特定 的 顺序 。 





输入 了 


SELECT prod_name 
FROM Products; 


输出 了 


prod_name 

Fish bean bag toy 
Bird bean bag toy 
Rabbit bean bag toy 
8 inch teddy bear 
12 inch teddy bear 
18 inch teddy bear 
Raggedy Ann 

King doll 

Queen doll 
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HX, 检索 出 的 数据 并 不 是 随机 显示 的 。 如 果 不 排序 ,数据 一 般 将 以 它 在 
底层 表 中 出 现 的 顺序 显示 ,这 有 可 能 是 数据 最 初 添加 到 表 中 的 顺序 。 但 是 ， 
如 果 数 据 随后 进行 过 更 新 或 删除 ， 那 么 这 个 顺序 将 会 受到 DBMS 重用 回 
收 存储 空间 的 方式 的 影响 。 因 此 ,如 果 不 明确 控制 的 话 , 则 最 终 的 结果 不 
能 (也 不 应 该 ) 依赖 该 排序 顺序 。 关 系数 据 库 设 计 理 论 认为 ,如 果 不 明确 
规定 排序 顺序 ， 则 不 应 该 假定 检索 出 的 数据 的 顺序 有 任何 意义 。 


























子 句 (clause ) 

SQL 语句 由 子 句 构 成， 有些 子 名 是 必需 的 ， 有 些 则 是 可 选 的 。 一 个 子 
句 通常 由 一 个 关键 字 加 上 所 提供 的 数据 组 成 。 子 句 的 例子 有 我 们 在 前 
一 课 看 到 的 SELECT 语句 的 FROM + éJ , 











为 了 明确 地 排序 用 SELECT 语句 检索 出 的 数据 ， 可 使 用 ORDER BY 子 句 。 
ORDER BY 子 句 取 一 个 或 多 个 列 的 名 字 ,， 据 此 对 输出 进行 排序 。 请 看 下 面 
的 例子 : 





输入 了 
SELECT prod_name 


FROM Products 
ORDER BY prod_name ; 


BY 


除了 指示 DBMS 软件 对 prod_name 列 以 字母 顺序 排序 数据 的 ORDER BY 
子 句 外 ， 这 条 语句 与 前 面 的 语句 相同 。 结 果 如 下 。 





输出 了 
prod_name 


12 inch teddy bear 
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18 inch teddy bear 
8 inch teddy bear 
Bird bean bag toy 
Fish bean bag toy 
King doll 

Queen doll 

Rabbit bean bag toy 
Raggedy Ann 





注意 : ORDER BY 子 句 的 位 置 
在 指定 一 条 ORDER BY 子 句 时 ， 应 该 保证 它 是 SELECT 语句 中 最 后 一 
条 子 句 。 如 果 它 不 是 最 后 的 子 句 ， 将 会 出 现 错误 消息 。 








提示 : 通过 非 选 择 列 进行 排序 
通常 ，ORDER BY 子 名 中 使 用 的 列 将 是 为 显示 而 选择 的 列 。 但 是 ， 实 
际 上 并 不 一 定 要 这 样 ， 用 非 检 索 的 列 排序 数据 是 完全 合法 的 。 











3.2 按 多 个 列 排序 


需要 按 不 止 一 个 列 进行 数据 排序 。 例 如 ， 如 果 要 显示 雇员 名 单 ， 可 
望 按 姓 和 名 排序 ( 首先 按 姓 排序 ， 然 后 在 每 个 姓 中 再 按 名 排序 )。 如 
多 个 雇员 有 相同 的 姓 ， 这 样 做 很 有 用 。 








常 
希 


N | At 
m S HN 








要 按 多 个 列 排序 ， 简 单 指定 列 名 ， 列 名 之 间 用 逗号 分 开 即 可 〈 就 像 选择 
多 个 列 时 那样 )。 








下 面 的 代码 检索 3 个 列 ， 并 按 其 中 两 个 列 对 结果 进行 排序 一 一 首先 按 价 
格 ， 然 后 按 名 称 排序 。 


输入 了 


SELECT prod_id，prod_price，prod_name 
FROM Products 
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ORDER BY prod_price, prod_name; 


输出 了 

prod_id prod_price prod_name 

BNBGO2 3.4900 Bird bean bag toy 
BNBGO1 3.4900 Fish bean bag toy 
BNBGO3 3.4900 Rabbit bean bag toy 
RGAN01 4.9900 Raggedy Ann 

BRO1 5.9900 8 inch teddy bear 
BRO2 8.9900 12 inch teddy bear 
RYLO1 9.4900 King doll 

RYLO2 9.4900 Queen doll 

BRO3 11.9900 18 inch teddy bear 


重要 的 是 理解 在 按 多 个 列 排序 时 ， 排 序 的 顺序 完全 按 规定 进行 。 换 句 话 
说 ， 对 于 上 述 例子 中 的 输出 ， 仅 在 多 个 行 具 有 相同 的 prod_price 值 时 
才 对 产品 按 prod_name 进行 排序 。 如 果 prod_price 列 中 所 有 的 值 都 是 
唯一 的 ， 则 不 会 按 prod_name 排序 。 





3.3” 按 列 位 置 排序 


除了 能 用 列 名 指出 排序 顺序 外 ，ORDER BY 还 支持 按 相 对 列 位 置 进行 排 
序 。 为 理解 这 一 内 容 ， 我 们 来 看 个 例子 : 





输入 了 


SELECT prod_id，prod_price，prod_name 
FROM Products 
ORDER BY 2, 3; 


输出 了 


prod_id prod_price prod_name 
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BNBG02 3.4900 Bird bean bag toy 
BNBGO1 3.4900 Fish bean bag toy 
BNBG03 3.4900 Rabbit bean bag toy 
RGAN01 4.9900 Raggedy Ann 

BRO1 5.9900 8 inch teddy bear 
BR02 8.9900 12 inch teddy bear 
RYL01 9.4900 King doll 

RYLO2 9.4900 Queen doll 

BRO3 11.9900 18 inch teddy bear 
SH Y 


可 以 看 到 ， 这 里 的 输出 与 上 面 的 查询 相同 ， 不 同 之 处 在 于 ORDER BY F 
句 。SELECT 清单 中 指定 的 是 选择 列 的 相对 位 置 而 不 是 列 名 。ORDER BY 2 
表示 按 SELECT 清单 中 的 第 二 个 列 prod_name 进行 排序 。ORDER BY 2， 
3 表示 先 按 prod_price， 再 按 prod_name 进行 排序 。 















































这 一 技术 的 主要 好 处 在 于 不 用 重新 输入 列 名 。 但 它 也 有 缺点 。 首先 , 不 明 
确 给 出 列 名 有 可 能 造成 错 用 列 名 排序 。 其 次 ,在 对 SELECT 清单 进行 更 改 
时 容易 错误 地 对 数据 进行 排序 (忘记 对 ORDER BY 子 句 做 相应 的 改动 ) 

最 后 ， 如 果 进 行 排序 的 列 不 在 SELECT 清单 中 ， 显 然 不 能 使 用 这 项 技术 。 














提示 : 按 非 选择 列 排序 
显然 ， 当 根据 不 出 现在 SELECT 清单 中 的 列 进行 排序 时 ， 不 能 采用 这 项 
技术 。 但是， 如 果 有 必要 ， 可 以 混合 使 用 实际 列 名 和 相对 列 位 置 。 











3.4 指定 排序 方向 


数据 排序 不 限于 升序 排序 ( 从 A #J Z ), 这 只 是 默认 的 排序 顺序 。 还 可 以 
使 用 ORDER BY 子 句 进行 降序 (从 Z 到 A) 排序 。 为 了 进行 降序 排序 ， 
必须 指定 DESC 关键 字 。 
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um 





下 面 的 例子 以 价格 降序 来 排序 产品 〈 最 贵 的 排 在 最 前 面 ): 


输入 了 


SELECT prod_id，prod_price，prod_name 
FROM Products 
ORDER BY prod_price DESC; 


输出 了 

prod_id prod_price prod_name 

BRO3 11.9900 18 inch teddy bear 
RYLO1 9.4900 King doll 

RYLO2 9.4900 Queen doll 

BRO2 8.9900 12 inch teddy bear 
BRO1 5.9900 8 inch teddy bear 
RGAN01 4.9900 Raggedy Ann 

BNBGO1 3.4900 Fish bean bag toy 
BNBG02 3.4900 Bird bean bag toy 
BNBG03 3.4900 Rabbit bean bag toy 








如 果 打 算 用 多 个 列 排序 ， 该 怎么 办 ? 下 面 的 例子 以 降序 排序 产品 ( 最 
的 在 最 前 面 )， 再 加 上 产品 名 : 

输入 了 

SELECT prod_id，prod_price，prod_name 


FROM Products 
ORDER BY prod_price DESC, prod_name; 


输出 了 

prod_id prod_price prod_name 

BRO3 11.9900 18 inch teddy bear 
RYLO1 9.4900 King doll 

RYLO2 9.4900 Queen doll 

BRO2 8.9900 12 inch teddy bear 
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BRO1 5.9900 8 inch teddy bear 
RGAN01 4.9900 Raggedy Ann 

BNBG02 3.4900 Bird bean bag toy 
BNBGO1 3.4900 Fish bean bag toy 
BNBG03 3.4900 Rabbit bean bag toy 
SH Y 














DESC 关键 字 只 应 用 到 直接 位 于 其 前 面 的 列 名 。 在 上 例 中 ,只 对 prod_price 
列 指定 DESC， 对 prod_name 列 不 指定 。 因 此 ，prod_price 列 以 降序 排 
J, mi prod_name 列 (在 每 个 价格 内 ) 仍然 按 标 准 的 升序 排序 。 











警告 : 在 多 个 列 上 降序 排序 
如 果 想 在 多 个 列 上 进行 降序 排序 ， 必 须 对 每 一 列 指定 DESC 关键 字 。 














请 注意 ,DESC 是 DESCENDING 的 缩写 ,这 两 个 关键 字 都 可 以 使 用 ,与 DESC 
相对 的 是 ASC ( 或 ASCENDING )， 在 升序 排序 时 可 以 指定 它 。 但 实际 上 ， 
ASC 没有 多 大 用 处 ， 因 为 升序 是 默认 的 〈 如 果 既 不 指定 ASC 也 不 指定 
DESC， 则 假定 为 ASC )。 











提示 : 区 分 大 小 写 和 排序 顺序 
在 对 文本 性 数据 进行 排序 时 ，A 与 a 相同 吗 ? a 位 于 B 之前， 还 是 Z 
之 后 ? 这 些 问题 不 是 理论 问题 ， 其 答案 取决 于 数据 库 的 设置 方式 。 


在 字典 〈dictionary ) 排序 顺序 中 ，A 被 视 为 与 a 相同 ， 这 是 大 多 数 数 
据 库 管理 系统 的 默认 行为 。 但 是 ,许多 DBMS 允许 数据 库 管 理 员 在 需 
要 时 改变 这 种 行为 (如果 你 的 数据 库 包含 大 量 外 语 字 符 ， 可 能 必须 这 
样 做 )。 


这 里 的 关键 问题 是 ,如 果 确 实 需 要 改变 这 种 排序 顺序 ,用 简单 的 ORDER 
BY 子 句 可 能 做 不 到 。 你 必须 请 求 数 据 库 管 理 员 的 帮助 。 
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35 ”小结 


这 一 课 学 习 了 如 何 用 SELECT 语句 的 ORDER BY 子 句 对 检索 出 的 数据 进行 
排序 。 这 个 子 句 必须 是 SELECT 语句 中 的 最 后 一 条 子 句 。 根 据 需 要 ， 可 
以 利用 它 在 一 个 或 多 个 列 上 对 数据 进行 排序 。 
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这 一 课 将 讲授 如 何 使 用 SELECT 语句 的 WHERE 子 句 指定 搜索 条 件 。 


4.1 使 用 WHERE F4 
数据 库 表 一 般 包 含 大 量 的 数据 ， 很 少 需要 检索 表 中 的 所 有 行 。 











通常 只 会 


根据 特定 操作 或 报告 的 需要 提取 表 数 据 的 子 集 。 只 检索 所 需 数据 需要 指 





定 搜 索 条 件 ( search criteria ), 搜索 条 件 也 称 为 过 滤 条 件 ( filter condition )。 


在 SELECT 语句 中 ， 数 据 根据 WHERE 子 句 中 指定 的 搜索 条 件 进行 过 滤 。 


WHERE 子 句 在 表 名 ( FROM FAJ) 之 后 给 出 ， 如 下 所 示 : 
输入 了 


SELECT prod_name，prod_price 


FROM Products 
WHERE prod_price = 3.49; 


BY 





这 条 语句 从 products 表 中 检索 两 个 列 ， 但 不 返回 所 有 行 ， 


prod_price 值 为 3.49 的 行 ， 如 下 所 示 : 


输出 了 


prod_name prod_price 
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Fish bean bag toy 3.49 
Bird bean bag toy 3.49 
Rabbit bean bag toy 3.49 





这 个 示例 使 用 了 简单 的 相等 检验 : 检查 这 一 列 的 值 是 否 为 指定 值 ， 据 此 
过 滤 数 据 。 不 过 ，SQL 不 止 能 测试 等 于 ， 还 能 做 更 多 的 事情 。 





提示 : 有 多 少 个 0? 

你 在 练习 这 个 示例 时 ， 会 发 现 显示 的 结果 可 能 是 3.49、3.490、3.4900 
等 。 出 现 这 样 的 情况 ,往往 是 因为 DBMS 指定 了 所 使 用 的 数据 类 型 及 
其 默认 行为 。 所 以 ,如果 你 的 输出 可 能 与 书 上 的 有 点 不 同 , TLE, 
毕竟 从 数学 角度 讲 ，3.49 和 3.4900 是 一 样 的 。 








提示 : SOL 过 滤 与 应 用 过 滤 

数据 也 可 以 在 应 用 层 过 滤 。 为 此 ，SQL 的 SELECT 语句 为 客户 端 应 用 
检索 出 超过 实际 所 需 的 数据 ， 然 后 客户 端 代码 对 返回 数据 进行 循环 ， 
提取 出 需要 的 行 。 


常 ， 这 种 做 法 极其 不 妥 。 优 化 数据 库 后 可 以 更 快速 有 效 地 对 数据 进 
过 滤 。 而 让 客户 端 应 用 (或 开发 语言 ) 处 理 数据 库 的 工作 将 会 极 大 
地 影响 应 用 的 性 能 ,并 且 使 所 创建 的 应 用 完全 不 具备 可 伸缩 性 。 此 外 ， 
如 果 在 客户 端 过 滤 数 据 ， 服 务 器 不 得 不 通过 网 络 发 送 多 余 的 数据 ， 这 
将 导致 网 络 带宽 的 浪费 。 


通 
行 








注意 : WHERE 子 句 的 位 置 

在 同时 使 用 ORDER BY 和 WHERE 子 名 时， 应 该 让 ORDER BY 位 于 
WHERE 之 后 ， 否 则 将 会 产生 错误 (关于 ORDER BY 的 使 用 ， 请 参阅 
第 3 课 )。 
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4.2 WHERE 子 句 操作 符 


我 们 在 做 相等 检验 时 看 到 了 第 一 个 WHERE 子 句 ， 它 确定 一 个 列 是 否 包含 
指定 的 值 。SQL 支持 表 4-1 列 出 的 所 有 条 件 操 作 符 。 





表 4-1 WHERE 子 句 操作 符 





























操作 符 说 “RB 操作 符 说 — HB 
x 等 于 $ KF 
<> 不 等 于 >= 大 于 等 于 
!= 不 等 于 1> 不 大 于 
< 小 于 BETWEEN 在 指定 的 两 个 值 之 间 
<= 小 于 等 于 IS NULL 为 NULL 值 
< 不 小 于 








注意 : ETRS 

表 4-1 中 列 出 的 某 些 操作 符 是 宛 余 的 (如 < > 与 1= 相 同 ，!< 相 当 于 >= )。 
并 非 所 有 DBMS 都 支持 这 些 操作 符 。 想 确定 你 的 DBMS 支持 哪些 操作 
符 ， 请 参阅 相应 的 文档 。 











4.2.1 检查 单个 值 








我 们 已 经 看 到 了 检验 相等 的 例子 , 现在 来 看 看 几 个 使 用 其 他 操作 符 的 例子 。 




















NL 





第 一 个 例子 是 列 出 所 有 价格 小 于 10 美元 的 产品 。 


输入 到 
SELECT prod_name, prod_price 


FROM Products 
WHERE prod_price < 10; 
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输出 


prod_name prod_price 
Fish bean bag toy 
Bird bean bag toy 
Rabbit bean bag toy 
8 inch teddy bear 
12 inch teddy bear 


O (O + OO Qn Ü ww wu 
Kel 
O 


Raggedy Ann 99 
King doll 49 
Queen doll 49 





下 一 条 语句 检索 所 有 价格 小 于 等 于 10 美元 的 产品 ( 因为 没有 价格 恰好 是 
10 美元 的 产品 ， 所 以 结果 与 前 一 个 例子 相同 ): 


输入 了 


SELECT prod_name，prod_price 
FROM Products 
WHERE prod_price <= 10; 


4.2.2 不 匹配 检查 
这 个 例子 列 出 所 有 不 是 供应 商 DLL01 制造 的 产品 : 





输入 了 


SELECT vend_id, prod_name 
FROM Products 
WHERE vend_id <> 'DLLO1'; 


输出 了 
vend_id prod_name 
BRS01 8 inch teddy bear 
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BRS01 12 inch teddy bear 
BRS01 18 inch teddy bear 
FNG01 King doll 
FNG01 Queen doll 





提示 : 何 时 使 用 引号 

如 果 仔细 观察 上 述 WHERE 子 名 中 的 条 件 ， 会 看 到 有 的 值 括 在 单 引 号 内 ， 
而 有 的 值 未 括 起 来 。 单 引号 用 来 限定 字符 串 。 如 果 将 值 与 字符 串 类 型 的 
列 进行 比较 ， 就 需要 限定 引号 。 用 来 与 数值 列 进行 比较 的 值 不 用 引号 。 
































下 面 是 相同 的 例子 ， 其 中 使 用 != 而 不 是 <> 操 作 符 : 
输入 了 


SELECT vend_id，prod_name 
FROM Products 
WHERE vend_id != 'DLLO1'; 





注意 : 是 != 还 是 <>? 


1!= 和 <> 通 常 可 以 互 换 , 但 是 , 并非 所 有 DBMS 都 支持 这 两 种 不 等 于 操 
作 符 。 例 如 ，Microsoft Access 支持 <> 而 不 支持 !=。 如 果 有 疑问 ， 请 参 
阅 相 应 的 DBMS 文档 。 











423 ”范围 值 检 查 








要 检查 某 个 范围 的 值 ， 可 以 使 用 BETWEEN 操作 符 。 其 语法 与 其 他 WHERE 
子 句 的 操作 符 稍 有 不 同 , 因为 它 需要 两 个 值 , 即 范围 的 开始 值 和 结束 值 。 
例如 , BETWEEN 操作 符 可 用 来 检索 价格 在 5 美元 和 10 美元 之 间 的 所 有 产 
品 ， 或 在 指定 的 开始 日 期 和 结束 日 期 之 间 的 所 有 日 期 。 











下 面 的 例子 说 明 如 何 使 用 BETWEEN 操作 符 ， 它 检索 价格 在 5 美元 和 10 
美元 之 间 的 所 有 产品 。 
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输入 了 


SELECT prod_name，prod_price 
FROM Products 
WHERE prod_price BETWEEN 5 AND 10; 


输出 了 

prod_name prod_price 
8 inch teddy bear 5.99 

12 inch teddy bear 8.99 

King doll 9.49 

Queen doll 9.49 
£V 


从 这 个 例子 可 以 看 到 , 在 使 用 BETWEEN 时 ， 必 须 指 定 两 个 值 一 一 所 需 范 
围 的 低 端 值 和 高 端 值 。 这 两 个 值 必须 用 AND 关键 字 分 隔 。BETWEEN 匹配 
范围 中 所 有 的 值 ， 包 括 指定 的 开始 值 和 结束 值 。 











42.4 FERE 


在 创建 表 时 ， 表 设计 人 员 可 以 指定 其 中 的 列 能 否 不 包含 值 。 在 一 个 列 不 
包含 值 时 ， 称 其 包含 空 值 NULL。 





NULL 
无 值 (no value )， 它 与 字段 包含 0、 空 字符 串 或 仅仅 包含 空格 不 同 。 














确定 值 是 否 为 NULL， 不 能 简单 地 检查 是 否 = NULL, SELECT 语句 有 一 个 
特殊 的 WHERE 子 句 ， 可 用 来 检查 具有 NULL 值 的 列 。 这 个 WHERE 子 句 就 
是 IS NULL 子 句 。 其 语法 如 下 : 











输入 了 


SELECT prod_name 
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FROM Products 
WHERE prod_price IS NULL; 

















这 条 语句 返回 所 有 没有 价格 〈 空 prod_price 字段 ， 不 是 价格 为 0 ) 的 

关 品 ， 由 于 表 中 没有 这 样 的 行 ， 所 以 没有 返回 数据 。 但 是 ，Customers 
表 确 实 包含 具有 NULL 值 的 列 : 如 果 没 有 电子 邮件 地 址 ， 则 cust_emai ] 
列 将 包含 NULL 值 : 








输入 了 


SELECT cust_name 
FROM CUSTOMERS 
WHERE cust_email IS NULL; 


输出 了 


CUSt_name 


Kids Place 
The Toy Store 





提示 : 各 DBMS 特有 的 操作 符 
许多 DBMS 扩展 了 标准 的 操作 符 集 ， 提 供 了 更 高 级 的 过 滤 选 择 。 更 多 
信息 请 参阅 相应 的 DBMS 文档 。 








注意 : NULL 和 非 匹 配 

通过 过 滤 选 择 不 包含 指定 值 的 所 有 行 时 ， 你 可 能 希望 返回 含 NULL 值 
的 行 。 但 是 这 做 不 到 。 因 为 未 知 (unknown ) 有 特殊 的 含义 ， 数 据 库 
不 知道 它们 是 否 匹配 ， 所 以 在 进行 匹配 过 滤 或 非 匹 配 过 滤 时 ， 不 会 返 
回 这 些 结果 。 


过 滤 数 据 时 ， 一 定 要 验证 被 过 滤 列 中 含 NULL 的 行 确实 出 现在 返回 的 
数据 中 。 
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4.3 小 结 


这 一 课 介绍 了 如 何 用 SELECT 语句 的 WHERE 子 句 过 滤 返 回 的 数据 。 我 
们 学 习 了 如 何 检 验 相 等 、 不 相等 、 大 于 、 小 于 、 值 的 范围 以 及 NULL 
值 等 。 
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第 5 课 高 级 数据 过 滤 








这 一 课 讲授 如 何 组 合 WHERE 子 句 以 建立 功能 更 强 、 更 高 级 的 搜索 条 件 。 
我 们 还 将 学 习 如 何 使 用 NOT 和 IN 操作 符 。 


5.1 组 合 WHERE FE 

各 4 课 介绍 的 所 有 WHERE 子 句 在 过 滤 数 据 时 使 用 的 都 是 单一 的 条 件 。 为 
了 进行 更 强 的 过 滤 控 制 ，SQL 人 允许 给 出 多 个 WHERE 子 句 。 这 些 子 句 有 两 
种 使 用 方式 ， 即 以 AND 子 句 或 OR 子 句 的 方式 使 用 。 




















yÈ 

















操作 符 ( operator ) 
用 来 联结 或 改变 WHERE 子 名 中 的 子 名 的 关键 字 ， 也 称 为 逻辑 操作 符 


(logical operator )。 











5.1.1 _ AND 操作 符 


要 通过 不 止 一 个 列 进行 过 滤 , 可 以 使 用 AND 操作 符 给 WHERE 子 句 附加 条 
件 。 下 面 的 代码 给 出 了 一 个 例子 : 





输入 了 


SELECT prod_id，prod_price，prod_name 
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FROM Products 
WHERE vend_id = 'DLL01' AND prod_price <= 4; 


BY 


此 SQL 语句 检索 由 供应 商 DLLOL 制造 旦 价格 小 于 等 于 4 美元 的 所 有 产品 
的 名 称 和 价格 。 这 条 SELECT 语句 中 的 WHERE 子 句 包含 两 个 条 件 , 用 AND 
关键 字 联 结 在 一 起 。AND 指示 DBMS 只 返回 满足 所 有 给 定 条 件 的 行 。 如 
果 某 个 产品 由 供应 商 DLL01 制造 ， 但 价格 高 于 4 美元 ， 则 不 检索 它 。 类 
似 地 ， 如 果 产 品 价格 小 于 4 美元 ,但 不 是 由 指定 供应 商 制 造 的 也 不 被 检 
索 。 这 条 SQL 语句 产生 的 输出 如 下 : 









































输出 

prod_id prod_price prod_name 

BNBGO2 3.4900 Bird bean bag toy 
BNBGO1 3.4900 Fish bean bag toy 
BNBGO3 3.4900 Rabbit bean bag toy 
AND 





用 在 WHERE 子 句 中 的 关键 字 ， 用 来 指示 检索 满足 所 有 给 定 条 件 的 行 。 














这 个 例子 只 包含 一 个 AND 子 句 ， 因 此 最 多 有 两 个 过 滤 条 件 。 可 以 增加 多 
个 过 滤 条 件 ， 每 个 条 件 间 都 要 使 用 AND 关键 字 。 











说 明 : 没有 ORDER BY FA 

为 了 节省 空间 ， 也 为 了 减少 你 的 输入 ， 我 在 很 多 例子 里 省 略 了 ORDER 
BY 子 句 。 因 此 ， 你 的 输出 完全 有 可 能 与 书 上 的 输出 不 一 致 。 虽 然 返 回 
行 的 数量 总 是 对 的 ， 但 它们 的 顺序 可 能 不 同 。 当 然 ， 如 果 你 愿意 也 可 
以 加 上 一 个 ORDER BY 子 句 ， 它 应 该 放 在 WHERE 子 名 之 后 。 
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5.1.2 ORR ETS 


OR 操作 符 与 AND 操作 符 正 好 相反 ， 它 指示 DBMS 检索 匹配 任 一 条 件 的 
行 。 事 实 上 ,许多 DBMS 在 OR WHERE 子 句 的 第 一 个 条 件 得 到 满足 的 情 
况 下 ， 就 不 再 计算 第 二 个 条 件 了 (在 第 一 个 条 件 满足 时 ， 不管 第 二 个 条 
件 是 否 满足 ， 相 应 的 行 都 将 被 检索 出 来 )。 

请 看 如 下 的 SELECT 语句 : 

输入 了 




















SELECT prod_name，prod_price 
FROM Products 
WHERE vend_id = "DLLO1L' OR vend_id = ‘BRSO1’; 


BY 





此 SQL 语句 检索 由 任 一 个 指定 供应 商 制造 的 所 有 产品 的 产品 名 和 价格 。 
OR 操作 符 告诉 DBMS 匹配 任 一 条 件 而 不 是 同时 匹配 两 个 条 件 。 如 果 这 里 
使 用 的 是 AND 操作 符 , 则 没有 数据 返回 ( 因为 会 创建 没有 匹配 行 的 WHERE 
TFA) 这 条 SQL 语句 产生 的 输出 如 下 : 














输出 了 

prod_name prod_price 
Fish bean bag toy 3.4900 
Bird bean bag toy 3.4900 
Rabbit bean bag toy 3.4900 

8 inch teddy bear 5.9900 

12 inch teddy bear 8.9900 

18 inch teddy bear 11.9900 
Raggedy Ann 4.9900 

OR 





WHERE 子 名 中 使 用 的 关键 字 ， 用 来 表示 检索 匹配 任 一 给 定 条 件 的 行 。 
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541.3 求 值 顺序 

WHERE 子 句 可 以 包含 任意 数目 的 AND 和 OR 操作 符 。 人 允许 两 者 结合 以 进 
行 复杂 、 高 级 的 过 滤 。 

但 是 ， 组 合 AND 和 OR 会 带 来 了 一 个 有 趣 的 问题 。 为 了 说 明 这 个 问题 ， 
来 看 一 个 例子 。 假 如 需要 列 出 价格 为 10 美元 及 以 上 ,有 上 且 由 DLLO1L 或 BRSO1 
制造 的 所 有 产品 。 下 面 的 SELECT 语句 使 用 组 合 的 AND 和 OR 操作 符 建立 
了 一 个 WHERE FAJ: 


输入 了 


SELECT prod_name，prod_price 

FROM Products 

WHERE vend_id = 'DLLO1' OR vend_id = "BRS01' 
AND prod_price >= 10; 














输出 了 

prod_name prod_price 
Fish bean bag toy 3.4900 
Bird bean bag toy 3.4900 
Rabbit bean bag toy 3.4900 

18 inch teddy bear 11.9900 
Raggedy Ann 4.9900 
SH Y 








请 看 上 面 的 结果 。 返 回 的 行 中 有 4 行 价格 小 于 10 美元 ， 显 然 ， 返 回 的 行 
未 按 预 期 的 进行 过 滤 。 为 什么 会 这 样 呢 ? 原因 在 于 求 值 的 顺序 。SQL ( 像 
多 数 语言 一 样 ) 在 处 理 OR 操作 符 前 ,优先 处 理 AND 操作 符 。 当 SQL 看 到 
上 述 WHERE 子 句 时 ， 它 理解 为 : 由 供应 商 BRS01 制造 的 价格 为 10 美元 以 
上 的 所 有 产品 ,以 及 由 供应 商 DLLOL 制造 的 所 有 产品 , 而 不 管 其 价格 如 何 。 
换 名 话说， 由 于 AND 在 求 值 过 程 中 优先 级 更 高 ， 操 作 符 被 错误 地 组 合 了 。 
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此 问题 的 解决 方法 是 使 用 圆 括号 对 操作 符 进 行 明确 分 组 。 请 看 下 面 的 
SELECT 语句 及 输出 : 


输入 了 





SELECT prod_name，prod_price 

FROM Products 

WHERE (vend_id = "DLLO1L' OR vend_id = "BRS01') 
AND prod_price >= 10; 


输出 了 
prod_name prod_price 
18 inch teddy bear — 11.99000 — 
Biy 











这 条 SELECT 语句 与 前 一 条 的 唯一 差别 是 ， 将 前 两 个 条 件 用 圆 括号 括 了 
起 来 。 因为 圆 插 号 具有 比 AND 或 OR 操作 符 更 高 的 求 值 顺序 , 所 以 DBMS 
首先 过 滤 圆 括号 内 的 OR 条 件 。. 这 时 ,SQL 语句 变 成 了 选择 由 供应 商 DLLO1 
或 BRS01 制造 的 且 价格 在 10 美元 及 以 上 的 所 有 产品 ， 这 正 是 我 们 希望 
的 结果 。 




















提示 : 在 WHERE 子 句 中 使 用 圆 括号 

任何 时 候 使 用 具有 AND 和 OR 操作 符 的 WHERE 子 句 ， 都 应 该 使 用 圆 括 
号 明确 地 分 组 操作 符 。 不 要 过 分 依赖 默认 求 值 顺序 ， 即 使 它 确实 如 你 
希望 的 那样 。 使 用 圆 括号 没有 什么 坏处 ， 它 能 消除 歧义 。 











5.2 IN 操作 符 


IN 操作 符 用 来 指定 条 件 范围 ， 范 围 中 的 每 个 条 件 都 可 以 进行 匹配 。IN 取 
一 组 由 逗号 分 隔 、 括 在 圆 括号 中 的 合法 值 。 下 面 的 例子 说 明了 这 个 操作 符 。 
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输入 了 


SELECT prod_name，prod_price 

FROM Products 

WHERE vend_id IN ( 'DLLO1', 'BRS01' ) 
ORDER BY prod_name; 


输出 了 

prod_name prod_price 
12 inch teddy bear 8.9900 

18 inch teddy bear 11.9900 

8 inch teddy bear 5.9900 
Bird bean bag toy 3.4900 
Fish bean bag toy 3.4900 
Rabbit bean bag toy 3.4900 
Raggedy Ann 4.9900 
分 株 


此 SELECT 语句 检索 由 供应 商 DLLOL 和 BRS01 制造 的 所 有 产品 ,IN 操作 
符 后 跟 由 逗号 分 隔 的 合法 值 ， 这 些 值 必须 括 在 圆 括号 中 。 























喜 你 猜 对 了 ! FI 


TH 


你 可 能 会 猜测 IN 操作 符 完成 了 与 OR 相同 的 功能 ， 均 
的 SQL 语句 完成 与 上 面 的 例子 相同 的 工作 。 





输入 了 


SELECT prod_name，prod_price 

FROM Products 

WHERE vend_id = "DLLO1L' OR vend_id = "BRS01' 
ORDER BY prod_name ; 


输出 


prod_name prod_price 
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12 inch teddy bear 8.9900 
18 inch teddy bear 11.9900 
8 inch teddy bear 5.9900 
Bird bean bag toy 3.4900 
Fish bean bag toy 3.4900 
Rabbit bean bag toy 3.4900 
Raggedy Ann 4.9900 


为 什么 要 使 用 IN 操作 符 ? 其 优点 如 下 。 


口 在 有 很 多 合法 选项 时 ，IN 操作 符 的 语法 更 清楚 ， 更 直观 。 

D 在 与 其 他 AND 和 OR 操作 符 组 合 使 用 IN 时 ， 求 值 顺 序 更 容易 管理 。 

D IN 操作 符 一 般 比 一 组 OR 操作 符 执行 得 更 快 ( 在 上 面 这 个 合法 选项 很 
少 的 例子 中 ， 你 看 不 出 性 能 差异 )。 

O IN 的 最 大 优点 是 可 以 包含 其 他 SELECT 语句 ， 能 够 更 动态 地 建立 
WHERE 子 句 。 第 11 课 会 对 此 进行 详细 介绍 。 

















一 


























IN 
WHERE 子 句 中 用 来 指定 要 匹配 值 的 清单 的 关键 字 ， 功 能 与 OR 相当 。 











5.3 NOT 操作 符 

WHERE 子 句 中 的 NOT 操作 符 有 且 只 有 一 个 功能 ， 那 就 是 否定 其 后 所 跟 的 
任何 条 件 。 因 为 NOT 从 不 单独 使 用 ( 它 总 是 与 其 他 操作 符 一 起 使 用 ), 所 
以 它 的 语法 与 其 他 操作 符 有 所 不 同 。NOT 关键 字 可 以 用 在 要 过 滤 的 列 前 ， 
而 不 仅 是 在 其 后 。 














NOT 
WHERE 子 句 中 用 来 否定 其 后 条 件 的 关键 字 。 














ERS 


aa 





下 面 的 例子 说 明 NOT 的 用 法 。 为 了 列 出 除 DLL01 之 外 的 所 有 供应 商 
的 产品 ， 可 编写 如 下 的 代码 。 
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输入 了 


SELECT prod_name 

FROM Products 

WHERE NOT vend_id = 'DLLO1' 
ORDER BY prod_name; 


输出 了 


prod_name 
12 inch teddy bear 
18 inch teddy bear 
8 inch teddy bear 
King doll 
Queen doll 


BY 























这 里 的 NOT 否定 跟 在 其 后 的 条 件 ， 因 此 ，DBMS 不 是 匹配 vendid 为 
DLLO1L， 而 是 匹配 非 DLL01 之 外 的 所 有 东西 。 


上 面 的 例子 也 可 以 使 用 <> 操 作 符 来 完成 ， 如 下 所 示 。 
输入 了 


SELECT prod_name 

FROM Products 

WHERE vend_id <> 'DLLO1' 
ORDER BY prod_name ; 

















输出 了 


prod_name 
12 inch teddy bear 
18 inch teddy bear 
8 inch teddy bear 
King doll 
Queen doll 
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BY 








为 什么 使 用 NOT? 对 于 这 里 的 这 种 简单 的 WHERE FAJ, 使 用 NOT 确实 
没有 什么 优势 。 但 在 更 复杂 的 子 句 中 ，NOT 是 非常 有 用 的 。 例 如 ， 在 
与 IN 操作 符 联 合 使 用 时 , NOT 可 以 非常 简单 地 找 出 与 条 件 列表 不 匹配 
的 行 。 





说 明 : MariaDB 中 的 NOT 
MariaDB 支持 使 用 NOT € £ IN. BETWEEN 和 EXISTS FA, K2% 
DBMS 允许 使 用 NOT 否定 任何 条 件 。 











54 小结 


这 一 课 讲 授 如 何 用 AND 和 OR 操作 符 组 合成 WHERE 子 句 ， 还 讲授 了 如 何 
明确 地 管理 求 值 顺序 ， 如 何 使 用 IN 和 NOT 操作 符 。 
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这 一 课 介 绍 什么 是 通配符 、 如 何 使 用 通配符 以 及 怎样 使 用 LIKE 操作 符 
进行 通 配 搜索 ， 以 便 对 数据 进行 复杂 过 滤 。 


6.1 LIKE 操作 符 

前 面 介绍 的 所 有 操作 符 都 是 针对 已 知 值 进行 过 滤 的 。 不 管 是 匹配 一 个 值 
还 是 多 个 值 ， 检 验 大 于 还 是 小 于 已 知 值 ， 或 者 检查 某 个 范围 的 值 ， 其 共 
同 点 是 过 滤 中 使 用 的 值 都 是 已 知 的 。 












































但 是 ， 这 种 过 滤 方 法 并 不 是 任何 时 候 都 好 用 。 例 如 ， 怎 样 搜索 产品 名 中 
包含 文本 bean bag 的 所 有 产品 ? 用 简单 的 比较 操作 符 肯 定 不 行 ， 必 须 使 
用 通配符 。 利 用 通配符 ， 可 以 创建 比较 特定 数据 的 搜索 模式 。 在 这 个 例 
子 中 ， 如 果 你 想 找 出 名 称 包含 bean bag 的 所 有 产品 ， 可 以 构造 一 个 通 配 
符 搜 索 模 式 ， 找 出 在 产品 名 的 任何 位 置 出 现 bean bag 的 产品 。 
































通配符 ( wildcard ) 
用 来 匹配 值 的 一 部 分 的 特殊 字符 。 








搜索 模式 ( search pattern ) 
由 字面 值 、 通 配 符 或 两 者 组 合 构成 的 搜索 条 件 。 
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通配符 本 身 实 际 上 是 SQL 的 WHERE 子 句 中 有 特殊 含义 的 字符 ,SQL 支持 


几 种 通配符 。 为 在 搜索 子 句 中 使 用 通配符 , 必须 使 用 LIKE 操作 符 。LIKE 
指示 DBMS, ， 后 跟 的 搜索 模式 利用 通配符 匹配 而 不 是 简单 的 相等 匹配 进 

















行 比较 。 





谓词 ( predicate ) 





操作 符 何 时 不 是 操作 符 ? 答案 是 ， 它 作为 谓词 时 。 从 技术 上 说 ，LIKE 
是 谓词 而 不 是 操作 符 。 虽 然 最 终 的 结果 是 相同 的 ， 但 应 该 对 此 术语 有 
所 了 解 ， 以 免 在 SQL 文献 或 手册 中 遇 到 此 术语 时 不 知 所 云 。 











通配符 搜索 只 能 用 于 文本 字段 (字符 串 ), 非 文本 数据 类 
通配符 搜索 。 














6.1.1 A25 (%) 通配符 


























型 字段 不 能 使 用 


最 常 使 用 的 通配符 是 百 分 号 (% )。 ERREP, % 表 示 任 何 字符 出 现任 意 次 
数 。 例 如 , 为 了 找 出 所 有 以 词 Fish 起 头 的 产品 , 可 发 布 以 下 SELECT 语句 : 


输入 了 


SELECT prod_id，prod_name 
FROM Products 
WHERE prod_name LIKE 'Fish%'; 


输出 了 

prod_id prod_name 

BNEGOL Fish bean bag toy 
SH Y 


此 例子 使 用 了 搜索 模式 'Fish%' 。 在 执行 这 条 子 句 时 
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， 将 检索 任意 以 
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Fish LHJ, % 告 诉 DBMS 接受 Fish 之 后 的 任意 字符 , 不 管 它 有 多 


少 字符 


公子 付 。 





说 明 : Access 通配符 
如 果 使 用 的 是 Microsoft Access， 需 要 使 用 * 而 不 是 %。 








说 明 : 区 分 大 小 写 
根据 DBMS 的 不 同 及 其 配置 ,搜索 可 以 是 区 分 大 小 写 的 。 如 果 区 分 大 
小 写 ， 则 'fish%' 与 Fish bean bag toy 就 不 匹配 。 














通配符 可 在 搜索 模式 中 的 任意 位 置 使 用 ， 并 且 可 以 使 用 多 个 通配符 。 下 
面 的 例子 使 用 两 个 通配符 ， 它 们 位 于 模式 的 两 端 : 








输入 了 


SELECT prod_id, prod_name 
FROM Products 
WHERE prod_name LIKE '%bean bag%'; 


输出 

prod_id prod_name 

BNBGO1 Fish bean bag toy 
BNBGO2 Bird bean bag toy 
BNBGO3 Rabbit bean bag toy 
SH Y 


搜索 模式 '%bean bag%' 表 示 匹 配 任何 位 置 上 包含 文本 bean bag 的 值 ， 
不 论 它 之 前 或 之 后 出 现 什么 字符 。 








通配符 也 可 以 出 现在 搜索 模式 的 中 间 ， 虽 然 这 样 做 不 太 有 用 。 下 面 的 例 
子 找 出 以 F 起 头 、 以 y 结尾 的 所 有 产品 : 


Tr 
fanny 
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输入 了 


SELECT prod_name 
FROM Products 
WHERE prod_name LIKE 'F%y'; 








提示 : 根据 部 分 信息 搜索 电子 邮件 地 址 

有 一 种 情况 下 把 通配符 放 在 搜索 模式 中 间 是 很 有 用 的 ， 就 是 根据 邮件 
地 址 的 一 部 分 来 查找 电子 邮件 ， 例 如 WHERE email LIKE 
"b%Qforta.com'。 











需要 特别 注意 ， 除 了 能 匹配 一 个 或 多 个 字符 外 ，% 还 能 匹配 0 个 字符 。% 
代表 搜索 模式 中 给 定位 置 的 0 个 、1 个 或 多 个 字符 。 











说 明 : 请 注意 后 面 所 跟 的 空格 

包括 Access 在 内 的 许多 DBMS 都 用 空格 来 填补 字段 的 内 容 。 例 如 , 如 
REIA 50 个 字符 ,而 存储 的 文本 为 Fish bean bag toy (17 个 字 
符 )， 则 为 填 满 该 列 需 要 在 文本 后 附加 33 个 空格 。 这 样 做 一 般 对 数据 
及 其 使 用 没有 影响 ,但 是 可 能 对 上 述 SQL 语句 有 负面 影响 。 子 多 WHERE 
prod_name LIKE 'F%y' 只 匹配 以 F 开头 、 以 y 结尾 的 prod_name。 

如 果 值 后 面 跟 空 格 ， 则 不 是 以 y 结尾 ， 所 以 Fish bean bag toy 就 
不 会 检索 出 来 ,简单 的 解决 办 法 是 给 搜索 模式 再 增加 一 个 % 号 : ' F%y%' 
还 匹配 y 之 后 的 字符 (或 空格 )。 更 好 的 解决 办 法 是 用 函数 去 掉 空 格 。 
请 参阅 第 8 课 。 











注意 : 请 注意 NULL 
通配符 % 看 起 来 像 是 可 以 匹配 任何 东西 ， 但 有 个 例外 ， 这 就 是 NULL, 
-F ë) WHERE prod_name LIKE '%' 不 会 匹配 产品 名 称 为 NULL 的 行 。 
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6.1.2 ”下划线 (_) 通配符 


另 一 个 有 用 的 通配符 是 下 划 线 (_) 下划线 的 用 途 与 % 一 样 , 但 它 只 匹配 
单个 字符 ， 而 不 是 多 个 字符 。 

















说 明 : DB2 通配符 
DB2 不 支持 通配符 _。 








说 明 : Access 通配符 
如 果 使 用 的 是 Microsoft Access， 需 要 使 用 ?而 不 是 _。 











举 一 个 例子 : 


输入 了 


SELECT prod_id，prod_name 
FROM Products 
WHERE prod_name LIKE '_ inch teddy bear ' ; 





说 明 : 请 注意 后 面 所 跟 的 空格 
与 上 例 一 样 ， 可 能 需要 给 这 个 模式 添加 一 个 通配符 。 











输出 了 

prod_id prod_name 

BRO2 12 inch teddy bear 
BRO3 18 inch teddy bear 
分 析 了 





这 个 WHERE 子 句 中 的 搜索 模式 给 出 了 后 面 跟 有 文本 的 两 个 通配符 。 结 
果 只 显示 匹配 搜索 模式 的 行 : 第 一 行 中 下 划 线 匹配 12， 第 二 行 中 匹配 
18。8 inch teddy bear 产品 没有 匹配 ， 因 为 搜索 模式 要 求 匹 配 两 个 
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通配符 而 不 是 一 个 。 对 照 一 下 ,下面 的 SELECT 语句 使 用 % 通 配 符 ， 返 
回 三 行 产品 : 








输入 了 


SELECT prod_id, prod_name 
FROM Products 
WHERE prod_name LIKE '% inch teddy bear'; 


输出 了 

prod_id prod_name 

BRO1 8 inch teddy bear 
BRO2 12 inch teddy bear 
BNR3 18 inch teddy bear 








与 % 能 匹配 0 个 字符 不 同 ，_ 总 是 刚好 匹配 一 个 字符 ， 不 能 多 也 不 能 少 。 


6.1.3 H5 (D 通配符 
方 括号 ( [] ) 通配符 用 来 指定 一 个 字符 集 ， 它 必须 匹配 指定 位 置 ( 通 配 
符 的 位 置 ) 的 一 个 字符 。 








说 明 : 并 不 总 是 支持 集合 

与 前 面 描述 的 通配符 不 一 样 ,并 不 是 所 有 DBMS 都 支持 用 来 创建 集合 
的 [] 。 只 有 微软 的 Access 和 SQL Server 支持 集合 。 为 确定 你 使 用 的 
DBMS 是 否 支 持 集合 ， 请 参阅 相应 的 文档 。 








例如 ， 找 出 所 有 名 字 以 J 或 M 起头 的 联系 人 ， 可 进行 如 下 查询 : 


输入 了 


SELECT cust_contact 
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FROM Customers 
WHERE cust_contact LIKE '[JM]%' 
ORDER BY cust_contact; 


输出 


cust_contact 


Jim Jones 
John Smith 
Michelle Green 


BY 


此 语句 的 WHERE 子 句 中 的 模式 为 ' [IJM]%' 。 这 一 搜索 模式 使 用 了 两 个 不 
同 的 通配符 。[JM] 匹 配方 括号 中 任意 一 个 字符 , 它 也 只 能 匹配 单个 字符 。 
因此 ,任何 多 于 一 个 字符 的 名 字 都 不 匹配 。[JM] 之 后 的 % 通 配 符 匹配 第 
一 个 字符 之 后 的 任意 数目 的 字符 ， 返 回 所 需 结果 。 




















此 通配符 可 以 用 前 缀 字符 ( 脱 字号 ) 来 否定 。 例 如 ， 下 面 的 查询 匹配 以 
J 和 M 之 外 的 任意 字符 起 头 的 任意 联系 人 名 《〈 与 前 一 个 例子 相反 ): 


输入 了 


SELECT cust_contact 

FROM Customers 

WHERE cust_contact LIKE '[ ° JM]%' 
ORDER BY cust_contact; 








说 明 : Access 中 的 否定 集合 
如 果 使 用 的 是 Microsoft Access， 需 要 用 1 而 不 是 ^ 来 否定 一 个 集合 ， 因 
此 ,使 用 的 是 [1JM] 而 不 是 [*JM]。 

















当然 , 也 可 以 使 用 NOT 操作 符 得 出 类 似 的 结果 。" 的 唯一 优点 是 在 使 用 多 
个 WHERE 子 句 时 可 以 简化 语法 : 
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输入 了 


SELECT cust_contact 

FROM Customers 

WHERE NOT cust_contact LIKE '[JM]%' 
ORDER BY cust_contact; 


6.2 ”使 用 通配符 的 技巧 

正如 所 见 ，SQL 的 通配符 很 有 用 。 但 这 种 功能 是 有 代价 的 ， 即 通配符 搜 
索 一 般 比 前 面 讨论 的 其 他 搜索 要 耗费 更 长 的 处 理 时 间 。 这 里 给 出 一 些 使 
用 通配符 时 要 记 住 的 技巧 。 











口 不 要 过 度 使 用 通配符 。 如 果 其 他 操作 符 能 达到 相同 的 目的 ， 应 该 使 用 
其 他 操作 符 。 

O 在 确实 需要 使 用 通配符 时 ， 也 尽量 不 要 把 它们 用 在 搜索 模式 的 开始 
处 。 把 通配符 置 于 开始 处 ， 搜 索 起 来 是 最 慢 的 。 
口 仔细 注意 通配符 的 位 置 。 如 果 放 错 地 方 ， 可 能 不 会 返回 想 要 的 数据 。 




































































总 之 ， 通 配 符 是 一 种 极其 重要 和 有 用 的 搜索 工具 ， 以 后 我 们 经 常会 用 
到 它 。 











63 小结 


这 一 课 介绍 了 什么 是 通配符 , 如 何在 WHERE 子 句 中 使 用 SQL 通配符 , 还 
说 明了 通配符 应 该 细心 使 用 ， 不 要 使 用 过 度 。 
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这 一 课 介绍 什么 是 计算 字段 ， 如 何 创 建 计 算 字 段 ， 以 及 如 何 从 应 用 程序 
中 使 用 别名 引用 它们 。 


71 计算 字段 
存储 在 数据 库 表 中 的 数据 一 般 不 是 应 用 程序 所 需要 的 格式 ， 下 面 举 几 个 
例子 。 











口 需要 显示 公司 名 ， 同 时 还 需要 显示 公司 的 地 址 ， 但 这 两 个 信息 存储 在 
不 同 的 表 列 中 。 

O 城市 、 州 和 邮政 编码 存储 在 不 同 的 列 中 ( 应 该 这 样 )， 但 邮件 标签 打 
印 程序 需要 把 它们 作为 一 个 有 恰当 格式 的 字段 检索 出 来 。 

口 列 数据 是 大 小 写 混合 的 ， 但 报表 程序 需要 把 所 有 数据 按 大 写 表 示 出 来 。 
O 物品 订单 表 存储 物品 的 价格 和 数量 ， 不 存储 每 个 物品 的 总 价格 ( 用 价 
格 乘 以 数量 即 可 )。 但 为 打印 发 票 ， 需 要 物品 的 总 价格 。 

口 需要 根据 表 数 据 进行 诸如 总 数 、 平 均 数 的 计算 。 












































在 上 述 每 个 例子 中 ， 存 储 在 表 中 的 数据 都 不 是 应 用 程序 所 需要 的 。 我 们 
需要 直接 从 数据 库 中 检索 出 转换 、 计 算 或 格式 化 过 的 数据 ， 而 不 是 检索 
出 数据 ， 然 后 再 在 客户 端 应 用 程序 中 重新 格式 化 。 
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这 就 是 计算 字段 可 以 派 上 用 场 的 地 方 了 了。 与 前 几 课 介绍 的 列 不 同 ， 计算 


字段 并 不 实际 存在 于 数据 库 表 中 。 计 算 字 段 是 运行 时 在 SELECT 语句 内 
创建 的 。 








字段 ( field ) 
基本 上 与 列 ( column ) 的 意思 相同 ， 经 常 互 换 使 用 ， 不 过 数据 库 列 一 








般 称 为 列 ， 而 术语 字段 通常 与 计算 字段 一 起 使 用 。 








需要 特别 注意 ， 只 有 数据 库 知道 SELECT 语句 中 哪些 列 是 实际 的 表 列 ， 
哪些 列 是 计算 字段 。 从 客户 端 ( 如 应 用 程序 ) 来 看 ， 计 算 字 段 的 数据 与 
其 他 列 的 数据 的 返回 方式 相同 。 








提示 : 客户 端 与 服务 器 的 格式 
在 SQL 语句 内 可 完成 的 许多 转换 和 格式 化 工作 都 可 以 直接 在 客户 端 
应 用 程序 内 完成 。 但 一 般 来 说 ， 在 数据 库 服 务 器 上 完成 这 些 操 作 比 在 








客户 端 中 完成 要 快 得 多 。 





7.2 ”拼接 字段 


为 了 说 明 如 何 使 用 计算 字段 ， 我 们 来 举 一 个 简单 例子 ， 创 建 由 两 列 组 成 
的 标题 。 





Vendors 表 包 含 供应 商 名 和 地 址 信息 。 假 如 要 生成 一 个 供应 商 报 表 ， 需 
要 在 格式 化 的 名 称 〈 位 置 ) 中 列 出 供应 商 的 位 置 。 





此 报表 需要 一 个 值 ， 而 表 中 数据 存储 在 两 个 列 vend_name 和 vend_ 
country 中 。 此 外 ， 需 要 用 括号 将 vend_country 括 起 来 ， 这 些 东 西 都 
没有 存储 在 数据 库 表 中 。 这 个 返回 供应 商 名 称 和 地 址 的 SELECT 语句 很 
简单 ， 但 我 们 是 如 何 创建 这 个 组 合 值 的 呢 ? 
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拼接 ( concatenate ) 
将 值 联结 到 一 起 ( 将 一 个 值 附 加 到 另 一 个 值 ) 构成 单个 值 。 








解决 办 法 是 把 两 个 列 拼 接 起 来 。 在 SQL 中 的 SELECT 语句 中 ， 可 使 用 一 
个 特殊 的 操作 符 来 拼接 两 个 列 。 根 据 你 所 使 用 的 DBMS ， 此 操作 符 可 用 
加 号 (+ ) 或 两 个 竖 杠 (|| ) 表示 。 在 MySQL 和 MariaDB 中 ， 必 须 使 用 
特殊 的 函数 。 

















说 明 : 是 + 还 是 |1? 
Access fe SQL Server 使 用 + 号 。DB2、Oracle、PostgreSQL 、SQLite 和 
Open Office Base 使 用 | |。 详 细 请 参阅 具体 的 DBMS 文档 。 





























下 面 是 使 用 加 号 的 例子 (多数 DBMS 使 用 这 种 语法 ): 


输入 了 


SELECT vend_name + ' (' + vend_country + ')' 
FROM Vendors 
ORDER BY vend_name; 


输出 了 

Bear Emporium (USA ) 
Bears R Us (USA ) 
Doll House Inc. (USA ) 
Fun and Games (England ) 
Furball Inc. (USA ) 
Jouets et ours (France 3) 

















下 面 是 相同 的 语句 ， 但 使 用 的 是 11 语法 : 





输入 了 


SELECT vend_name || ' (' |] vend_country || ')' 
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FROM Vendors 
ORDER BY vend_name; 


输出 了 

Bear Emporium (USA ) 
Bears R Us (USA ) 
Doll House Inc. (USA ) 
Fun and Games (England ) 
Furball Inc. (USA ) 
Jouets et ours (France ) 

















下 面 是 使 用 MySQL 或 MariaDB 时 需要 使 用 的 语句 : 
输入 了 


SELECT Concat(vend_name, ' (', vend_country, ')') 
FROM Vendors 
ORDER BY vend_name; 


Sr YV 
上 面 两 个 SELECT 语句 拼接 以 下 元 素 : 


O 存储 在 vend_name 列 中 的 名 字 ; 

口 包含 一 个 空格 和 一 个 左 圆 括号 的 字符 串 ; 
口 存储 在 vend_country 列 中 的 国家 ; 

口 包含 一 个 右 圆 括号 的 字符 串 。 
从 上 述 输出 中 可 以 看 到 ，SELECT 语句 返回 包含 上 述 四 个 元 素 的 一 个 列 
(计算 字段 )。 

再 看 看 上 述 SELECT 语句 返回 的 输出 。 结 合成 一 个 计算 字段 的 两 个 列 用 
空格 填充 。 许 多 数据 库 ( 不 是 所 有 ) 保存 填充 为 列 宽 的 文本 值 ， 而 实际 
上 你 要 的 结果 不 需要 这 些 空格 。 为 正确 返回 格式 化 的 数据 ， 必 须 去 掉 这 
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些 空格 。 这 可 以 使 用 SQL 的 RTRIMO 函数 来 完成 ， 如 下 所 示 : 
输入 了 


SELECT RTRIM(vend_name) + ' (' + RTRIM(vend_country) + ')' 
FROM Vendors 
ORDER BY vend_name; 


Bear Emporium (USA) 
Bears R Us (USA) 

Doll House Inc. (USA) 
Fun and Games (England) 
Furball Inc. (USA) 
Jouets et ours (France) 


下 面 是 相同 的 语句 ， 但 使 用 的 是 | |: 




















输入 了 


SELECT RTRIM(vend_name) || ' (' || RTRIM(vend_country) || ')' 
FROM Vendors 
ORDER BY vend_name; 


Bear Emporium (USA) 
Bears R Us (USA) 

Doll House Inc. (USA) 
Fun and Games (England) 
Furball Inc. (USA) 
Jouets et ours (France) 


BY 























RTRIMO 函数 去 掉 值 右边 的 所 有 空格 。 通 过 使 用 RTRIMO ， 各 个 列 都 进 
行 了 整理 。 
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说 明 : TRIM 函数 

大 多 数 DBMS 都 支持 RTRIMO ( 正如 刚才 所 见 ， 它 去 掉 字 符 串 右边 的 
空格 )、LTRIM()〈 去 掉 字 符 串 左边 的 空格 ) 以 及 TRIMO (去 掉 字符 
串 左右 两 边 的 空格 )。 











使 用 别名 


从 前 面 的 输出 可 以 看 到 ，SELECT 语句 可 以 很 好 地 拼接 地 址 字段 。 但 是 ， 
这 个 新 计算 列 的 名 字 是 什么 呢 ? 实际 上 它 没 有 名 字 ， 它 只 是 一 个 值 。 如 
果 仅 在 SQL 查询 工具 中 查看 一 下 结果 ， 这样 没 有 什么 不 好 。 但是， 一 个 
未 命名 的 列 不 能 用 于 客户 端 应 用 中 ， 因 为 客户 端 没 有 办 法 引用 它 。 












































为 了 解决 这 个 问题 ，SQL 支持 列 别名 。 别 名 (alias ) 是 一 个 字段 或 值 的 
替换 名 。 别 名 用 As 关键 字 赋 予 。 请 看 下 面 的 SELECT 语句 : 











输入 了 


SELECT RTRIM(vend_name) + ' (' + RTRIM(vend_country) + ')' 
AS vend_title 

FROM Vendors 

ORDER BY vend_name; 


输出 


vend_title 

Bear Emporium (USA) 
Bears R Us (USA) 

Doll House Inc. (USA) 
Fun and Games (England) 
Furball Inc. (USA) 
Jouets et ours (France) 


下 面 是 相同 的 语句 ， 但 使 用 的 是 11 语 法 : 
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输入 了 


SELECT RTRIM(vend_name) || ' (' || RTRIM(vend_country) || ')' 
AS vend_title 

FROM Vendors 

ORDER BY vend_name; 

















下 面 是 MySQL 和 MariaDB 中 使 用 的 语句 : 





输入 了 


SELECT Concat(vend_name, ' (', vend_country, ')') 
AS vend_title 

FROM Vendors 

ORDER BY vend_name; 


BY 


SELECT 语句 本 身 与 以 前 使 用 的 相同 , 只 不 过 这 里 的 计算 字段 之 后 跟 了 文 
本 AS vend_title。 它 指示 SQL 创建 一 个 包含 指定 计算 结果 的 名 为 
vend_title 的 计算 字段 。 从 输出 可 以 看 到 ， 结 果 与 以 前 的 相同 ， 但 现 
在 列 名 为 vend_title, 任何 客户 端 应 用 都 可 以 按 名 称 引 用 这 个 列 , 就 像 
它 是 一 个 实际 的 表 列 一 样 。 























说 明 : AS 通常 可 选 
在 很 多 DBMS 中 ，AS 关键 字 是 可 选 的 ， 不 过 最 好 使 用 它 ， 这 被 视 为 
一 条 最 佳 实践 。 








提示 : 别名 的 其 他 用 途 
别名 还 有 其 他 用 途 。 常 见 的 用 途 包 括 在 实际 的 表 列 名 包含 不 合法 的 字 
符 (如 空格 ) 时 重新 命名 它 ， 在 原来 的 名 字 含 混 或 容易 误解 时 扩充 它 。 
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注意 : 别名 

别名 的 名 字 既 可 以 是 一 个 单词 ， 也 可 以 是 一 个 字符 串 。 如 果 是 后 者 ， 
字符 串 应 该 括 在 引号 中 。 虽然 这 种 做 法 是 合法 的 , 但 不 建议 这 么 去 做 。 
多 单词 的 名 字 可 读 性 高 ， 不 过 会 给 客户 端 应 用 带 来 各 种 问题 。 因 此 ， 
别名 最 常见 的 使 用 是 将 多 个 单词 的 列 名 重 命名 为 一 个 单词 的 名 字 。 








说 明 : 导出 列 
别名 有 时 也 称 为 导出 列 ( derived column ), 不管 怎么 叫 ， 它 们 所 代表 
的 是 相同 的 东西 。 











73 ”执行 算术 计算 

计算 字段 的 另 一 常见 用 途 是 对 检索 出 的 数据 进行 算术 计算 。 举 个 例子 ， 
Orders 表 包 含 收 到 的 所 有 订单 ，OrderItems 表 包 含 每 个 订单 中 的 各 项 
物品 。 下 面 的 SQL 语句 检索 订单 号 20008 中 的 所 有 物品 : 








输入 了 


SELECT prod_id, quantity, item_price 
FROM OrderItems 
WHERE order_num = 20008; 


输出 了 

prod_id quantity item_price 
RGAN01 5 4.9900 
BRO3 5 11.9900 
BNBG01 10 3.4900 
BNBG02 10 3.4900 
BNBG03 10 3.4900 


item_price 列 包 含 订单 中 每 项 物品 的 单价 。 如 下 汇总 物品 的 价格 É 
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建 计算 字段 
价 乘 以 订购 数量 ): 


输入 了 


SELECT prod_id， 

quantity, 

item_price, 

quantity*item_price AS expanded_price 
FROM OrderItems 
WHERE order_num = 20008; 


输出 了 

prod_id quantity item_price expanded_price 
RGAN01 5 4.9900 24.9500 

BRO3 5 11.9900 59.9500 

BNBG01 10 3.4900 34.9000 

BNBG02 10 3.4900 34.9000 

BNBG03 10 3.4900 34.9000 

分 株 


输出 中 显示 的 expanded_price 列 是 一 个 计算 字段 , 此 计算 为 quantity* 
item_price。 客 户 端 应 用 现在 可 以 使 用 这 个 新 计算 列 ， 就 像 使 用 其 他 列 
一 样 。 

SQL 支持 表 7-1 中 列 出 的 基本 算术 操作 符 。 此 外 ， 圆 括号 可 用 来 区 分 优 
先 顺序 。 关 于 优先 顺序 的 介绍 ， 请 参阅 第 5 课 。 




















表 7-1 SQL 算 木 操作 符 





操作 符 说 明 
+ 加 
; w 
s R 
/ 除 
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提示 : 如 何 测试 计算 

SELECT 语句 为 测试 .检验 函数 和 计算 提供 了 很 好 的 方法 ,虽然 SELECT 
通常 用 于 从 表 中 检索 数据 , 但 是 省 略 了 FROM 子 句 后 就 是 简单 地 访问 和 
处 理 表达 式 ， 例 如 SELECT 3 * 2; 将 返回 6，SELECT Trim(' abc '); 
将 返回 abc, SELECT Now() ;使 用 Now() 函 数 返回 当前 日 期 和 时 间 。 
现在 你 明白 了 ， 可 以 根据 需要 使 用 SELECT 语句 进行 检验 。 











7.4 小 结 

这 一 课 介绍 了 计算 字段 以 及 如 何 创 建 计算 字段 。 我 们 用 例子 说 明了 计算 
字段 在 字符 串 拼接 和 算术 计算 中 的 用 途 。 此 外 ， 还 讲述 了 如 何 创建 和 使 
用 别名 ， 以 便 应 用 程序 能 引用 计算 字段 。 
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这 一 课 介 绍 什么 是 函数 , DBMS 支持 何 种 函数 ,以 及 如 何 使 用 这 些 函 数 ; 
还 将 讲解 为 什么 SQL 函数 的 使 用 可 能 会 带 来 问题 。 














8.1 函数 


与 大 多 数 其 他 计算 机 语言 一 样 ，SQL 也 可 以 用 函数 来 处 理 数据 。 函 数 一 
般 是 在 数据 上 执行 的 ， 为 数据 的 转换 和 处 理 提供 了 方便 。 








前 一 课 中 用 来 去 掉 字 符 串 尾 的 空格 的 RTRIMO 就 是 一 个 函数 。 


函数 带 来 的 问题 


在 学 习 这 一 课 并 进行 实践 之 前 ， 你 应 该 了 解 使 用 SQL 函数 所 存在 的 





与 几乎 所 有 DBMS 都 等 同 地 支持 SQL 语句 ( 如 SELECT) 不 同 ， 每 一 个 
DBMS 都 有 特定 的 函数 .事实 上 , 只 有 少数 几 个 函数 被 所 有 主要 的 DBMS 
等 同 地 支持 。 虽 然 所 有 类 型 的 函数 一 般 都 可 以 在 每 个 DBMS 中 使 用 , 但 
各 个 函数 的 名 称 和 语法 可 能 极其 不 同 。 为 了 说 明 可 能 存在 的 问题 , 表 8-1 
列 出 了 3 个 第 用 的 函数 及 其 在 各 个 DBMS 中 的 语法 : 
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表 8-1 DBMS 824893 Fe 








































































































A #⁄ W 法 
提取 字符 串 的 组 成 ”Access 使 用 MIDGO ; DB2 、Oracle PostgreSQL 和 SQLite 使 用 
部 分 SUBSTR(); MySQL 和 SQL Server 使 用 SUBSTRING() 
数据 类 型 转换 Access 和 Oracle 使 用 多 个 函数 ,每 种 类 型 的 转换 有 一 个 函数 ; DB2 
和 PostgreSQL 使 用 CASTO); MariaDB 、MySQL 和 SQL Server 使 用 
CONVERT) 
取 当 前 日 期 Access fli H] NOWO ; DB2 和 PostgreSQL 使 用 CURRENT_DATE ; 



































MariaDB 和 MySQL 使 用 CURDATE() ; Oracle 使 用 SYSDATE; SQL 
Server 使 用 GETDATE() ;SQLite 使 用 DATEO) 
































可 以 看 到 , 与 SQL 语句 不 一 样 ，SQL 也 数 不 是 可 移植 的 。 这 意味 着 为 特 
定 SQL 实现 编写 的 代码 在 其 他 实现 中 可 能 不 正常 。 


























可 移植 ( portable ) 
所 编写 的 代码 可 以 在 多 个 系统 上 运行 











为 了 代码 的 可 移植 , 许多 SQL 程序 员 不 赞成 使 用 特定 于 实现 的 功能 。 虽 
然 这 样 做 很 有 好 处 ， 但 有 的 时 候 并 不 利于 应 用 程序 的 性 能 。 如 果 不 使 用 
这 些 函数 ， 编 写 某 些 应 用 程序 代码 会 很 艰难 。 必 须 利 用 其 他 方法 来 实现 
DBMS 可 以 非常 有 效 完 成 的 工作 。 






































提示 : 是 否 应 该 使 用 函数 ? 

现在 ， 你 面临 是 否 应 该 使 用 函数 的 选择 。 决 定 权 在 你 ， 使 用 或 是 不 使 
用 也 没有 对 错 之 分 。 如 果 你 决定 使 用 函数 ， 应 该 保证 做 好 代码 注释 ， 
以 便 以 后 你 (或 其 他 人 ) 能 确切 地 知道 所 编写 的 SQL 代码 的 含义 。 











8.2 ”使 用 函数 
大 多 数 SQL 实现 支持 以 下 类 型 的 限 数 。 
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O 用 于 处 理 文本 字符 串 〈 如 删除 或 填充 值 ， 转 换 值 为 大 写 或 小 写 ) 的 文 
本 函数 。 

口 用 于 在 数值 数据 上 进行 算术 操作 ( 如 返回 绝对 值 ， 进 行 代数 运算 ) 的 
数值 函数 。 

O 用 于 处 理 日 期 和 时 间 值 并 从 这 些 值 中 提取 特定 成 分 (如 返回 两 个 日 期 
之 差 ， 检 查 日 期 有 效 性 ) 的 日 期 和 时 间 函 数 。 

O 返回 DBMS 正 使 用 的 特殊 信息 〈 如 返回 用 户 登 录 信息 ) 的 系统 函数 。 























我 们 在 上 一 课 看 到 函数 用 于 SELECT 后 面 的 列 名 ， 但 函数 的 作用 不 仅 于 
此 。 它 还 可 以 作为 SELECT 语句 的 其 他 成 分 ， 如 在 WHERE 子 句 中 使 用 ， 
在 其 他 SQL 语句 中 使 用 等 ， 后 面 会 做 更 多 的 介绍 。 


8.2.1 文本 处 理 函 数 


在 上 一 课 ， O 其 中 使 用 RTRIMO 
函数 来 去 除 列 值 右 边 的 空格 。 下 面 是 另 一 个 例子 ,这 次 使 用 的 是 UPPERO 
PRŽI: 

















输入 了 


SELECT vend_name, UPPER(vend_name) AS vend_name_upcase 
FROM Vendors 
ORDER BY vend_name; 


输出 了 

vend_name vend_name_upcase 
Bear Emporium BEAR EMPORIUM 
Bears R Us BEARS R US 

Doll House Inc. DOLL HOUSE INC. 
Fun and Games FUN AND GAMES 
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Furball Inc. FURBALL INC. 
Jouets et ours JOUETS ET OURS 


可 以 看 到 ，UPPERGO 将 文本 转换 为 大 写 ， 因 此 本 例子 中 每 个 供应 商都 列 
出 两 次 ， 第 一 次 为 Vendors 表 中 存储 的 值 ， 第 二 次 作为 列 vend_name_ 
upcase 转换 为 大 写 。 








K 8-2 列 出 了 一 些 常用 的 文本 处 理 函 数 。 


表 8-2 ”常用 的 文本 处 理 函 数 











































































































% 数 说 BB 
LEFTO (或 使 用 子 字符 串 函 数 ) 返回 字符 串 左 边 的 字符 
LENGTH()( 也 使 用 DATALENGTHQO 或 LENO ) 返回 字符 串 的 长 度 
LOWERO ( Access 使 用 LCASEC) ) 将 字符 串 转 换 为 小 写 
LTRIMO) 去 掉 字 符 串 左边 的 空格 
RIGHTO (或 使 用 子 字 符 串 函数 ) 返回 字符 串 右 边 的 字符 
RTRIMO 去 掉 字 符 串 右边 的 空格 
SOUNDEXO) 返回 字符 串 的 SOUNDEX 值 
UPPER()( Access 使 用 UCASEC) ) 将 字符 串 转换 为 大 写 

















表 8-2 中 的 SOUNDEX 需要 做 进一步 的 解释 。SOUNDEX 是 一 个 将 任何 文 
本 串 转换 为 描述 其 语音 表示 的 字母 数字 模式 的 算法 。SOUNDEX 考虑 了 
类 似 的 发 音字 符 和 音节 ， 使 得 能 对 字符 串 进行 发 音 比较 而 不 是 字母 比 
较 。 虽 然 SOUNDEX 不 是 SQL 概念 ， 但 多 数 DBMS 都 提供 对 SOUNDEX 
的 支持 。 






































说 明 : SOUNDEX 支持 
Microsoft Access 和 PostgreSQL 不 支持 SOUNDEX() ， 因 此 以 下 的 例子 
不 适用 于 这 些 DBMS。 
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AI, 如果 在 创建 SQLite 时 使 用 了 SQLITE_SOUNDEX 编译 时 选项 ， 那 
Z SOUNDEXO Æ SQLite 中 就 可 用 。 因 为 SQLITE_SOUNDEX 不 是 默认 
的 编译 时 选项 ， 所 以 多 数 SQLite 实现 不 支持 SOUNDEX O 。 





下 面 给 出 一 个 使 用 SOUNDEXO 函数 的 例子 。Customers 表 中 有 一 个 顾客 
Kids Place, HIX% Michelle Green。 但 如 果 这 是 错误 的 输入 ， 
此 联系 名 实际 上 应 该 是 Mi chae1 Green， 该 怎么 办 呢 ? 显然 ， 按 正确 的 
联系 名 搜索 不 会 返回 数据 ， 如 下 所 示 : 























输入 了 


SELECT cust_name, cust_contact 
FROM Customers 


WHERE cust_contact = 'Michael Green'; 
输出 了 
cust_name cust_contact 








现在 试 一 下 使 用 SOUNDEXO 函数 进行 搜索 ， 它 匹配 所 有 发 音 类 似 于 
Michael Green 的 联系 名 : 

输入 

SELECT cust_name, cust_contact 


FROM Customers 
WHERE SOUNDEX(cust_contact) = SOUNDEX('Michael Green'); 


输出 了 
cust_name cust_contact 
Kids Place Michelle Green 
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BY 


在 这 个 例子 中 ，WHERE 子 句 使 用 SOUNDEX O 函数 把 cust contact 列 值 
和 搜索 字符 串 转 换 为 它们 的 SOUNDEX 值 。 因 为 Michael Green 和 
Michelle Green 发 音 相似 ， 所 以 它们 的 SOUNDEX 值 匹配 ， 因 此 WHERE 
子 句 正确 地 过 滤 出 了 所 需 的 数据 。 





8.2.2 ”日 期 和 时 间 处 理 函 数 


日 期 和 时 间 采 用 相应 的 数据 类 型 存储 在 表 中 , 每 种 DBMS 都 有 自己 的 特 
丈 形式 。 日 期 和 时 间 值 以 特殊 的 格式 存储 ， 以 便 能 快速 和 有 效 地 排序 或 
过 滤 ， 并 且 方 省 物理 存储 空间 。 


























应 用 程序 一 般 不 使 用 日 期 和 时 间 的 存储 格式 ， 因 此 日 斯 和 时 间 消 数 总 是 
用 来 读 取 、 统 计 和 处 理 这 些 值 。 由 于 这 个 原因 ， 日 期 和 时 间 函 数 在 SQL 
中 具有 重要 的 作用 。 遗 贼 的 是 ， 它 们 很 不 一 致 ， 可 移植 性 最 差 。 


我 们 举 个 简单 的 例子 , 来 说 明日 期 处 理 函 数 的 用 法 。0rders 表 中 包含 的 
订单 都 带 有 订单 日 期 。 为 在 SQL Server 中 检索 2012 年 的 所 有 订单 , 可 如 
下 进行 : 





输入 

SELECT order_num 

FROM Orders 

WHERE DATEPART(yy, order_date) = 2012; 
输出 了 


order_num 
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20007 
20008 
20009 


在 Access 中 使 用 如 下 版 本 : 





输入 了 


SELECT order_num 
FROM Orders 
WHERE DATEPART('yyyy', order_date) = 2012; 


BY 


这 个 例子 (SQL Server 和 Sybase 版 本 以 及 Access 版 本 ) 使 用 了 DATEPARTO 
函数 ， 顾名思义， 此 函数 返回 日 期 的 某 一 部 分 。DATEPART() 函数 有 两 个 
参数 ， 它 们 分 别 是 返回 的 成 分 和 从 中 返回 成 分 的 日 期 。 在 此 例子 中 ， 
DATEPART() 只 从 order_date 列 中 返回 年 份 。 通 过 与 2012 比较 , WHERE 
子 句 只 过 滤 出 此 年 份 的 订单 。 


下 面 是 使 用 名 为 DATE_PARTO) 的 类 似 函 数 的 PostgreSQL 版 本 : 
输入 了 


SELECT order_num 
FROM Orders 
WHERE DATE_PART('year', order_date) = 2012; 












































Oracle 没有 DATEPARTO 函数 , 不 过 有 几 个 可 用 来 完成 相同 检索 的 日 期 处 
理 函 数 。 例 如 : 








输入 
SELECT order_num 


FROM Orders 
WHERE to_number(to_char(order_date, 'YYYY')) = 2012; 
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SH Y 

在 这 个 例子 中 ，to_char(0) 函数 用 来 提取 日 期 的 成 分 ，to_numberG) 用 
来 将 提取 出 的 成 分 转换 为 数值 ， 以 便 能 与 2012 进行 比较 。 

完成 相同 工作 的 男 一 方法 是 使 用 BETWEEN 操作 符 : 

输入 

SELECT order_num 

FROM Orders 


WHERE order_date BETWEEN to_date('01-01-2012') 
AND to_date('12-31-2012'); 























BY 


在 此 例子 中 ，Oracle 的 to_date O 函数 用 来 将 两 个 字符 串 转 换 为 日 期 。 
一 个 包含 2012 年 1 月 1 日 ， 另 一 个 包含 2012 年 12 月 31 H. BETWEEN 
操作 符 用 来 找 出 两 个 日 期 之 间 的 所 有 订单 。 值 得 注意 的 是 ， 相 同 的 代码 
在 SQL Server 中 不 起 作用 ， 因 为 它 不 支持 to_date0) 函数 。 但 是 ， 如 果 
用 CONVERTG) 蔡 换 to_date() ， 当 然 可 以 使 用 这 种 类 型 的 语句 。 
MySQL 和 MariaDB 具有 各 种 日 期 处 理 函 数 ， 但 没有 DATEPARTO o 
MySQL 和 MariaDB 用 户 可 使 用 名 为 YEARO 〇 的 函数 从 日 期 中 提取 年 份 : 
输入 

SELECT order_num 


FROM Orders 
WHERE YEAR(order_date) = 2012; 



























































在 SQLite 中 有 个 小 技巧 : 
输入 了 


SELECT order_num 
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FROM Orders 














函数 处 理 数据 








WHERE strftime('%Y', order_date) = '2012'; 


这 里 给 出 的 例子 提取 和 使 月 





同 的 处 理 ， 使 用 AND 操作 符 可 以 进行 年 和 月 份 的 比较 。 





日 期 的 成 分 (年 )。 按 月 份 过 滤 ， 可 以 进行 相 


DBMS 提供 的 功能 远 不 止 简单 的 日 期 成 分 提取 。 大 多 数 DBMS 具有 比较 
日 期 、 执 行 基于 日 期 的 运算 、 选 择 日 期 格式 等 的 函数 。 但 是 ， 
不 同 DBMS 的 日 期 -时 间 处 理 函 数 可 能 不 同 。 关 于 具体 DBMS 支持 的 日 





























期 -时 间 处 理 函 数 ， 请 参阅 相应 的 文档 。 


8.2.3 ”数值 处 理 函 数 
数值 处 理 函 数 仅 处 理 数值 数据 。 这 些 函数 一 般 主 要 用 于 代数 、 三 角 或 几 








可 以 看 到 ， 





何 运算 ， 因 此 不 像 字符 串 或 日 期 -时 间 处 理 函 数 使 用 那么 频繁 。 








具有 讽刺 意味 的 是 ,在 主要 DBMS 的 函数 中 ， 数 值 函 数 是 最 一 致 、 最 统 
一 的 函数 。 表 8-3 列 出 一 些 常 用 的 数值 处 理 函 数 。 


表 8-3 ”常用 数值 处 理 函 数 



































函 数 说 BB 

ABSO 返回 一 个 数 的 绝对 值 
coso 返回 一 个 角度 的 余弦 
EXPO) 返回 一 个 数 的 指数 值 
PIO) 返回 圆周 率 

SINO 返回 一 个 角度 的 正弦 
SQRTO) 返回 一 个 数 的 平方 根 
TANO 返回 一 个 角度 的 正切 






































关于 具体 DBMS 所 支持 的 算术 处 理 函 数 ， 请 参阅 相应 的 文档 。 
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8.3 小 结 


这 一 课 介 绍 了 如 何 使 用 SQL 的 数据 处 理 函 数 。 虽 然 这 些 函 数 在 格式 化 、 
处 理 和 过 滤 数 据 中 非常 有 用 ， 但 它们 在 各 种 SQL 实现 中 很 不 一 致 (SQL 
Server 和 Oracle 之 间 的 差异 说 明了 这 一 点 )。 


eni 
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这 一 课 介 绍 什么 是 SQL 的 聚集 函数 ， 如 何 利 用 它们 汇总 表 的 数据 。 
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我 们 经 常 需要 汇总 数据 而 不 用 把 它们 实际 检索 出 来 , 为 此 SQL 提供 了 专 
门 的 函数 。 使 用 这 些 函 数 ，SQL 查询 可 用 于 检索 数据 ， 以 便 分 析 和 报表 
生成 。 这 种 类 型 的 检索 例子 有 : 

口 确定 表 中 行 数 (或 者 满足 某 个 条 件 或 包含 某 个 特定 值 的 行 数 ); 

口 获得 表 中 某 些 行 的 和 ; 

口 找 出 表 列 〈 或 所 有 行 或 某 些 特定 的 行 ) 的 最 大 值 、 最 小 值 、 平 均值 。 
上 述 例子 都 需要 汇总 表 中 的 数据 ， 而 不 需要 实际 数据 本 身 。 因 此 ， 返 回 
实际 表 数 据 纯 属 浪费 时 间 和 人 处理 资源 ( 更 不 用 说 带宽 了 )。 再 说 一 遍 , 我 
们 实际 想 要 的 是 汇总 信息 。 















































为 方便 这 种 类 型 的 检索 ,SQL 给 出 了 5 个 聚集 函数 ， 见 表 9-1。 这 些 函 数 
能 进行 上 述 检索 。 与 前 一 章 介 绍 的 数据 处 理 函 数 不 同 ，SQL 的 聚集 函数 
在 各 种 主要 SQL 实现 中 得 到 了 相当 一 致 的 文 持 。 








聚集 函数 ( aggregate function ) 
对 某 些 行 运行 的 函数 ， 计 算 并 返回 一 个 值 。 
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函 数 说 了 明 

AVGO 返回 某 列 的 平均 值 

COUNTO 返回 某 列 的 行 数 

MAXO 返回 某 列 的 最 大 值 

MINO 返回 某 列 的 最 小 值 

SUMO 返回 某 列 值 之 和 
下 面 说 明 各 函数 的 使 用 。 


9.1.1 _ AVGO 5828 

AVG 〇 通过 对 表 中 行 数 计数 并 计算 其 列 值 之 和 ， 求 得 该 列 的 平均 值 。 AVGO 
可 用 来 返回 所 有 列 的 平均 值 ， 也 可 以 用 来 返回 特定 列 或 行 的 平均 值 。 
下 面 的 例子 使 用 AvGGO 返 回 Products 表 中 所 有 产品 的 平均 价格 : 
输入 了 


SELECT AVG(prod_price) AS avg_price 
FROM Products; 


输出 了 

avg_price 

6.823333 

Biy 

此 SELECT 语句 返回 值 avg_price， 它 包含 Products 表 中 所 有 产品 的 
平均 价格 。 如 第 7 课 所 述 ，avg_price 是 一 个 别名 。 








AVGO 也 可 以 用 来 确定 特定 列 或 行 的 平均 值 。 下 面 的 例子 返回 特定 供应 
商 所 提供 产品 的 平均 价格 : 


图 灵 社 区 会 员 maik000 专 享 尊重 版 权 








76 | 第 9 课 汇总 数据 


输入 了 


SELECT AVG(prod_price) AS avg_price 
FROM Products 
WHERE vend_id = 'DLLO1'; 


输出 


avg_price 


这 条 SELECT 语句 与 前 一 条 的 不 同 之 处 在 于 ， 它 包含 了 WHERE 子 句 。 此 
WHERE 子 句 仅 过 滤 出 vend_id 为 DLL01 的 产品 ， 因 此 avg_price 中 返 
回 的 值 只 是 该 供应 商 产品 的 平均 值 。 








注意 : 只 用 于 单个 列 
AVG() 只 能 用 来 确定 特定 数值 列 的 平均 值 ， 而 且 列 名 必须 作为 函数 参 
数 给 出 。 为 了 获得 多 个 列 的 平均 值 ， 必 须 使 用 多 个 AVG() 函数 。 








说 明 : NULL 值 
AVGO 函数 忽略 列 值 为 NULL 的 行 。 











9.1.2 COUNT O 函数 


COUNT C) 函数 进行 计数 。 可 利用 COUNT0) 确 定 表 中 行 的 数目 或 符合 特定 
条 件 的 行 的 数目 。 





COUNT O 函数 有 两 种 使 用 方式 : 








O 使 用 CoUNT(*) 对 表 中 行 的 数目 进行 计数 ， 不 管 表 列 中 包含 的 是 空 值 
(NULL ) 还 是 非 空 值 。 
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口 使 用 COUNTCcolumn) 对 特定 列 中 具有 值 的 行进 行 计数 ,忽略 NULL 值 。 
下 面 的 例子 返回 Customers 表 中 顾客 的 总 数 : 


输入 了 


SELECT COUNT(*) AS num_cust 
FROM Customers; 


输出 了 


num_cust 


BY 


在 此 例子 中 , 利用 COUNT(C*) 对 所 有 行 计数 , 不 管 行 中 各 列 有 什么 值 。 计 
数值 在 num_cust 中 返回 。 





下 面 的 例子 只 对 具有 电子 邮件 地 址 的 客户 计数 : 


输入 了 


SELECT COUNT(cust_email) AS num_cust 
FROM Customers; 


输出 


num_cust 


这 条 SELECT 语句 使 用 COUNT(cust_emai1) 对 cust_email 列 中 有 值 的 
行进 行 计数 。 在 此 例子 中 ，cust_email 的 计数 为 3 (表示 5 个 顾客 中 只 
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有 3 个 顾客 有 电子 邮件 地 址 )。 








说 明 : NULL 值 
如 果 指 定 列 名 ， 则 COUNT() 函数 会 忽略 指定 列 的 值 为 空 的 行 ， 但 如 果 
COUNT 函数 中 用 的 是 星 号 (* )， 则 不 忽略 。 











9.1.3 MAX( 函数 
MAXGO 返 回 指定 列 中 的 最 大 值 。MAX() 要求 指定 列 名 ， 如 下 所 示 : 
mA Y 


SELECT MAX(prod_price) AS max_price 
FROM Products; 


输出 了 
max_price 


11.9900 
BY 


这 里 ，MAX() 返 回 Products 表 中 最 贵 物品 的 价格 。 





提示 : 对 非 数值 数据 使 用 MAXO 

虽然 MAXO 〇 一 般 用 来 找 出 最 大 的 数值 或 日 期 值 ， 但 许多 (并非 所 有 ) 
DBMS 允许 将 它 用 来 返回 任意 列 中 的 最 大 值 ， 包 括 返 回 文 本 列 中 的 最 
大 值 。 在 用 于 文本 数据 时 ，MAX() 返 回 按 该 列 排序 后 的 最 后 一 行 。 








说 明 : NULL 值 
MAX() 函数 忽略 列 值 为 NULL 的 行 。 
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9.1.4 MINO 824 


MINC) 的 功能 正好 与 MAXO 功能 相反 , 它 返 回 指定 列 的 最 小 值 。 与 MAXO 
一 样 ，MINQO) 要 求 指定 列 名 ， 如 下 所 示 : 


输入 了 


SELECT MINCprod_price) AS min_price 
FROM Products; 


输出 了 


min_price 


其 中 MINORII Products 表 中 最 便宜 物品 的 价格 。 





提示 : 对 非 数值 数据 使 用 MINO 

虽然 MINO 〇 一 般 用 来 找 出 最 小 的 数值 或 日 期 值 ， 但 许多 (并非 所 有 ) 
DBMS 允许 将 它 用 来 返回 任意 列 中 的 最 小 值 ， 包 括 返 回 文 本 列 中 的 最 
小 值 。 在 用 于 文本 数据 时 ，MIN() 返 回 该 列 排序 后 最 前 面 的 行 。 








说 明 : NULL 值 
MINO 函数 忽略 列 值 为 NULL 的 行 。 











9.1.5 SUMO AŽ 
SUMQ) 用 来 返回 指定 列 值 的 和 (总计 )。 


下 面 举 一 个 例子 ，OrderItems 包含 订单 中 实际 的 物品 ， 每 个 物品 有 相 
应 的 数量 。 可 如 下 检索 所 订购 物品 的 总 数 (所 有 quantity 值 之 和 ): 
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输入 

SELECT SUM(quantity) AS items_ordered 
FROM OrderItems 

WHERE order_num = 20005; 

输出 了 


items_ordered 


BY 


因数 SUMCquantity) 返 回 订单 中 所 有 物品 数量 之 和 ，WHERE 子 句 保 订 
统计 某 个 物品 订单 中 的 物品 。 


m 





SUMO 也 可 以 用 来 合计 计算 值 。 在 下 面 的 例子 中 ， 人 合计 每 项 物品 的 
item_pricexquantity， 得 出 总 的 订单 金额 : 
输入 了 


SELECT SUM(item price*quantity) AS total_price 
FROM OrderItems 
WHERE order_num = 20005; 


输出 


total_price 


1648.0000 
BY 


PK Št SUMCitem_pricex*quantity) 返 回 订 单 中 所 有 物品 价钱 之 和 ， 
WHERE 子 句 同样 保证 只 统计 某 个 物品 订单 中 的 物品 。 
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提示 : 在 多 个 列 上 进行 计算 
如 本 例 所 示 ， 利 用 标准 的 算术 操作 符 ， 所 有 聚集 函数 都 可 用 来 执行 多 
个 列 上 的 计算 。 








说 明 : NULL 值 
SUM() 函数 忽略 列 值 为 NULL 的 行 。 








92 ”聚集 不 同 值 
以 上 5 个 聚集 函数 都 可 以 如 下 使 用 。 














口 对 所 有 行 执行 计算 ,指定 ALL 参数 或 不 指定 参数 ( 因为 ALL 是 默认 行 
为 )。 
口 只 包含 不 同 的 值 ， 指 定 DISTINCT 参数 。 











提示 : ALL 为 默认 
ALL 参数 不 需要 指定 ， 因 为 它 是 默认 行为 。 如 果 不 指定 DISTINCT, Mj 
假定 为 ALL。 








说 明 : 不 要 在 Access 中 使 用 

Microsoft Access 在 聚集 函数 中 不 支持 DISTINCT， 因 此 下 面 的 例子 不 
适合 于 Access。 要 在 Access 得 到 类 似 的 结果 ， 需 要 使 用 子 查询 把 
DISTINCT 数据 返回 到 外 部 SELECT COUNT(*) 语 句 。 














下 面 的 例子 使 用 AVGO 函数 返回 特定 供应 商 提供 的 产品 的 平均 价格 。 它 
与 上 面 的 SELECT 语句 相同 , 但 使 用 了 DISTINCT 参数 , 因此 平均 值 只 考 
虑 各 个 不 同 的 价格 : 
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输入 到 
SELECT AVG(DISTINCT prod_price) AS avg_price 


FROM Products 
WHERE vend_id = 'DLLO1'; 


输出 


avg_price 


可 以 看 到 , 在 使 用 了 DISTINCT 后 ， 此 例子 中 的 avg_price 比较 高 ， 
为 有 多 个 物品 具有 相同 的 较 低 价格 。 排 除 它 们 提升 了 平均 价格 。 











注意 : DISTINCT 不 能 用 于 COUNT(*) 
如 果 指 定 列 名 ， 则 DISTINCT 只 能 用 于 COUNT()。DISTINCT 不 能 
于 COUNT(C*)。 类 似 地 ，DISTINCT 必须 使 用 列 名 , 不 能 用 于 计算 或 表 








提示 : 将 DISTINCT HF MINO FI MAXO 

虽然 DISTINCT 从 技术 上 可 用 于 MIN() 和 MAX()， 但 这 样 做 实际 上 没 
有 价值 。 一 个 列 中 的 最 小 值 和 最 大 值 不 管 是 否 只 考虑 不 同 值 ， 结 果 都 
是 相同 的 。 








说 明 : 其 他 聚集 参数 

除了 这 里 介绍 的 DISTINCT 和 ALL 参数 ,有 的 DBMS 还 支持 其 他 参数 ， 
如 支持 对 查询 结果 的 子 集 进行 计算 的 TOP 和 TOP PERCENT。 为 了 解 具 
体 的 DBMS 支持 哪些 参数 ， 请 参阅 相应 的 文档 。 
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93 ”组 合 聚 集 函 数 


目前 为 止 的 所 有 聚集 函数 例子 都 只 涉及 单个 函数 。 但 实际 上 ,SELECT 语 
句 可 根据 需要 包含 多 个 肾 集 函数 。 请 看 下 面 的 例子 : 


输入 了 


SELECT COUNT(*) AS num_items ， 
MINCprod_price) AS price_min， 
MAX(prod_price) AS price_max, 
AVG(prod_price) AS price_avg 

FROM Products; 











输出 了 
num_items price_min price_max price_avg 
9 390 ao 6.823333. 
Biy 





这 里 用 单条 SELECT 语句 执行 了 4 个 聚集 计算 ， 返 回 4 个 值 ( Products 
表 中 物品 的 数目 ， 产 品 价格 的 最 高 值 、 最 低 值 以 及 平均 值 )。 





注意 : 取 别 名 

在 指定 别名 以 包含 菜 个 聚集 函数 的 结果 时 ， 不 应 该 使 用 表 中 实际 的 列 
名 。 虽 然 这 样 做 也 算 合 法 , 但 许多 SQL 实现 不 支持 ， 可 能 会 产生 模糊 
的 错误 消息 。 











94 小 结 

RERA RCA. SQL 支持 5 个 聚集 函数 ， 可 以 用 多 种 方法 使 用 
它们 ， 返回 所 需 的 结果 。 这 些 函 数 很 高 效 ， 它 们 返回 结果 一 般 比 你 在 自 
己 的 客户 端 应 用 程序 中 计算 要 快 得 多 。 
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这 一 课 介 绍 如 何 分 组 数据 ， 以 便 汇总 表 内 容 的 子 集 。 这 涉及 两 个 新 
SELECT 语句 子 句 : GROUP BY 子 句 和 HAVING 子 句 。 





10.1 数据 分 组 


从 上 一 课 得 知 ， 使 用 SQL 聚集 函数 可 以 汇总 数据 。 这 样 ， 我 们 就 能 够 
对 行进 行 计 数 ， 计 算 和 与 平均 数 ， 不 检索 所 有 数据 就 获得 最 大 值 和 最 
小 值 。 











目前 为 止 的 所 有 计算 都 是 在 表 的 所 有 数据 或 匹配 特定 的 WHERE 子 句 的 数 
据 上 进行 的 。 比 如 下 面 的 例子 返回 供应 商 DLL01 提供 的 产品 数目 : 





输入 了 

SELECT COUNT(*) AS num_prods 
FROM Products 

WHERE vend_id = 'DLLO1'; 


输出 了 


num_prods 
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如 果 要 返回 每 个 供应 商 提供 的 产品 数目 ， 该 怎么 办 ?或 者 返回 只 提供 一 
项 产品 的 供应 商 的 产品 ， 或 者 返回 提供 10 个 以 上 产品 的 供应 商 的 产品 ， 
怎么 办 ? 


这 就 是 分 组 大 显 身 手 的 时 候 了 。 使 用 分 组 可 以 将 数据 分 为 多 个 逻辑 组 ， 
对 每 个 组 进行 聚集 计算 。 





10.2 创建 分 组 


分 组 是 使 用 SELECT 语句 的 GROUP BY 子 句 建立 的 。 理解 分 组 的 最 好 办 法 
是 看 一 个 例子 : 

















输入 了 


SELECT vend_id, COUNT(*) AS num_prods 
FROM Products 
GROUP BY vend_id; 


输出 了 

vend_id num_prods 
BRS01 3 

DLL01 4 

FNG01 2 

Biy 


上 面 的 SELECT 语句 指定 了 两 个 列 : vendid 包含 产品 供应 商 的 ID, 
num_prods 为 计算 字段 (用 COUNT(*) 函 数 建立 )。GROUP BY 子 句 指示 
DBMS 按 vend_id 排序 并 分 组 数据 。 这 就 会 对 每 个 vend_id 而 不 是 整个 
表 计 算 num_prods 一 次 。 从 输出 中 可 以 看 到 ， 供 应 商 BRS01 有 3 个 产 
品 ， 供 应 商 DLLOL 有 4 个 产品 ， 而 供应 商 FNG01 有 2 个 产品 。 
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因为 使 用 了 GROUP BY， 就 不 必 指 定 要 计算 和 估 值 的 每 个 组 了 。 系 统 会 自 
动 完 成 。GROUP BY 子 句 指示 DBMS 分 组 数据 ， 然 后 对 每 个 组 而 不 是 整 
个 结果 集 进 行 聚集 。 











在 使 用 GROUP BY 子 句 前 ， 需 要 知道 一 些 重要 的 规定 。 





Q GROUP BY 子 句 可 以 包含 任意 数目 的 列 ， 因 而 可 以 对 分 组 进行 谍 套 ， 

更 细致 地 进行 数据 分 组 。 

口 如 果 在 GROUP BY 子 句 中 舱 套 了 分 组 ， 数 据 将 在 最 后 指定 的 分 组 上 进 
行 汇总 。 换 名 话说， 在 建立 分 组 时 ， 指 定 的 所 有 列 都 一 起 计算 ( 所 以 
不 能 从 个 别 的 列 取 回 数据 )。 

口 GROUP BY 子 句 中 列 出 的 每 一 列 都 必须 是 检索 列 或 有 效 的 表达 式 (但 
不 能 是 聚集 函数 ) 如 果 在 SELECT 中 使 用 表达 式 , 则 必须 在 GROUP BY 
子 句 中 指定 相同 的 表达 式 。 不 能 使 用 别名 。 

O 大 多 数 SQL 实现 不 允许 GROUP BY 列 带 有 长 度 可 变 的 数据 类 型 ( 如 文 

本 或 备注 型 字段 )。 

O 除 聚 集 计 算 语句 外 ，SELECT 语句 中 的 每 一 列 都 必须 在 GROUP BY FAJ 

中 给 出 。 

口 如 果 分 组 列 中 包含 具有 NULL 值 的 行 ， 则 NULL 将 作为 一 个 分 组 返回 。 

如 果 列 中 有 多 行 NULL 值 ， 它 们 将 分 为 一 组 。 

口 GROUP BY 子 句 必须 出 现在 WHERE 子 句 之 后 ，ORDER BY 子 句 之 前 。 



































提示 : ALLFA 

Microsoft SQL Server 等 有 些 SQL 实现 在 GROUP BY 中 支持 可 选 的 ALL 
子 句 。 这 个 子 句 可 用 来 返回 所 有 分 组 ， 即 使 是 没有 匹配 行 的 分 组 也 返 
回 (在 此 情况 下 ， 聚 集 将 返回 NULL )。 具 体 的 DBMS 是 否 支 持 ALL, 
请 参阅 相应 的 文档 。 
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注意 : 通过 相对 位 置 指定 列 

有 的 SQL 实现 允许 根据 SELECT 列表 中 的 位 置 指定 GROUP BY 的 列 。 

例如 ，GROUP BY 2, 1 可 表示 按 选 择 的 第 二 个 列 分 组 ， 然 后 再 按 第 一 
个 列 分 组 。 虽 然 这 种 速记 语法 很 方便 ， 但 并 非 所 有 SQL 实现 都 支持 ， 
并 且 使 用 它 容 易 在 编辑 SQL 语句 时 出 错 。 








10.3 过 滤 分 组 

除了 能 用 GROUP BY 分 组 数据 外 ，SQL 还 允许 过 滤 分 组 ， 规 定 包括 哪些 
分 组 ， 排 除 哪 些 分 组 。 例 如 ， 你 可 能 想 要 列 出 至 少 有 两 个 订单 的 所 有 顾 
客 。 为 此 ， 必 须 基 于 完整 的 分 组 而 不 是 个 别 的 行进 行 过 滤 。 

















我 们 已 经 看 到 了 WHERE 子 句 的 作用 (第 4 课 提 及 )。 但 是 ， 在 这 个 例子 
中 WHERE 不 能 完成 任务 ,因为 WHERE 过 滤 指 定 的 是 行 而 不 是 分 组 。 事 实 
上 ，WHERE 没有 分 组 的 概念 。 




















那么 ,不 使 用 WHERE 使 用 什么 呢 ? SQL 为 此 提供 了 另 一 个 子 句 ， 就 是 
HAVING J. HAVING 非常 类 似 于 WHERE。 事 实 上 ， 目 前 为 止 所 学 过 的 
所 有 类 型 的 WHERE 子 句 都 可 以 用 HAVING 来 替代 。 唯 一 的 差别 是 , WHERE 
过 滤 行 ， 而 HAVING 过 滤 分 组 。 























提示 : HAVING 支持 所 有 WHERE 操作 符 

在 第 4 课 和 第 5 课 中 ,我 们 学 习 了 WHERE 子 句 的 条 件 (包括 通配符 条 
件 和 带 多 个 操作 符 的 子 句 )。 学 过 的 这 些 有 关 WHERE 的 所 有 技术 和 选 
项 都 适用 于 HAVING。 它 们 的 句法 是 相同 的 ， 只 是 关键 字 有 差别 。 








那么 ,怎么 过 滤 分 组 呢 ? 请 看 以 下 的 例子 : 
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输入 了 


SELECT cust_id, COUNT(*) AS orders 
FROM Orders 

GROUP BY cust_id 

HAVING COUNT(*) >= 2; 


输出 了 

Cust_id orders 
1000000001 2 
SH Y 











这 条 SELECT 语句 的 前 三 行 类 似 于 上 面 的 语句 。 最 后 一 行 增加 了 HAVING 
子 句 ， 它 过 滤 COUNT(*) >= 2 (两 个 以 上 订单 ) 的 那些 分 组 。 
可 以 看 到 ，WHERE 子 句 在 这 里 不 起 作用 ， 因 为 过 滤 是 基于 分 组 聚集 值 ， 
而 不 是 特定 行 的 值 。 












































说 明 : HAVING 和 WHERE 的 差别 

这 里 有 另 一 种 理解 方法 , WHERE 在 数据 分 组 前 进行 过 滤 ，HAVING 在 数 
据 分 组 后 进行 过 滤 。 这 是 一 个 重要 的 区 别 ，WHERE 排除 的 行 不 包括 在 
分 组 中 。 这 可 能 会 改变 计算 值 ， 从 而 影响 HAVING 子 句 中 基于 这 些 值 
过 滤 掉 的 分 组 。 














qlip 





那么 , 有 没有 在 一 条 语句 中 同时 使 用 WHERE 和 HAVING 子 句 的 需要 呢 ? 事实 
上 ， 确 实 有 。 假 如 想 进一步 过 滤 上 面 的 语句 ， 使 它 返回 过 去 12 个 月 内 具有 
两 个 以 上 订单 的 顾客 。 为 此 ， 可 增加 一 条 WHERE 子 句 ， 过 滤 出 过 去 12 个 月 
内 下 过 的 订单 ， 然 后 再 增加 HAVING 子 句 过 滤 出 具有 两 个 以 上 订单 的 分 组 。 























为 了 更 好 地 理解 ， 来 看 下 面 的 例子 ， 它 列 出 具有 两 个 以 上 产品 且 其 价格 
大 于 等 于 4 的 供应 商 : 
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输入 了 


SELECT vend_id, COUNT(*) AS num_prods 
FROM Products 

WHERE prod_price >= 4 

GROUP BY vend_id 

HAVING COUNT(*) >= 2; 


输出 了 

vend_id num_prods 
s 3 
FNG01 2 

Biy 








这 条 语句 中 , 第 一 行 是 使 用 了 聚集 函数 的 基本 SELECT 语句 ,很 像 前 面 的 
例子 。WHERE 子 句 过 滤 所 有 prod_price 至 少 为 4 的 行 , 然后 按 vend_id 
分 组 数据 ，HAVING 子 句 过 滤 计 数 为 2 或 2 以 上 的 分 组 。 如 果 没 有 WHERE 
子 句 ， 就 会 多 检索 出 一 行 (供应 商 DLLOL， 销 售 4 个 产品 ， 价 格 都 在 4 
以 下 ): 














输入 了 


SELECT vend_id, COUNT(*) AS num_prods 
FROM Products 

GROUP BY vend_id 

HAVING COUNT(*) >= 2; 


输出 了 

vend_id num_prods 
BRS01 3 

DLL01 4 

FNG01 2 
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说 明 : 使 用 HAVING 和 WHERE 

HAVING 与 WHERE 非常 类 似 ， 如 果 不 指 定 GROUP BY， 则 大 多 数 DBMS 
会 同等 对 待 它 们 。 不 过 ， 你 自己 要 能 区 分 这 一 点 。 使 用 HAVING 时 应 
该 结合 GROUP BY F4, mm WHERE 子 名 用 于 标准 的 行 级 过 滤 。 








10.4 “分 组 和 排序 


GROUP BY 和 ORDER BY 经 常 完成 相同 的 工作 ， 但 它们 非常 不 同 ， 理 解 这 
一 点 很 重要 。 表 10-1 汇总 了 它们 之 间 的 差别 。 


表 10-1 ORDER BY 与 GROUP BY 











































































































ORDER BY GROUP BY 
对 产生 的 输出 排序 对 行 分 组 ， 但 输出 可 能 不 是 分 组 的 顺序 
任意 列 都 可 以 使 用 ( 甚至 非 ”只 可 能 使 用 选择 列 或 表达 式 列 ,而 且 必须 使 用 每 个 选择 列 
选择 的 列 也 可 以 使 用 ) 表达 式 
不 一 定 需要 如 果 与 聚集 函数 一 起 使 用 列 〈 或 表达 式 ) ， 则 必须 使 








KR 10-1 中 列 出 的 第 一 项 差别 极为 重要 。 我 们 经 常 发 现 ， 用 GROUP BY 分 
组 的 数据 确实 是 以 分 组 顺序 输出 的 。 但 并 不 总 是 这 样 ， 这 不 是 SQL 规范 
所 要 求 的 。 此 外 ， 即 使 特定 的 DBMS 总 是 按 给 出 的 GROUP BY 子 句 排序 
数据 ， 用 户 也 可 能 会 要 求 以 不 同 的 顺序 排序 。 就 因为 你 以 某 种 方式 分 组 
数据 ( 获得 特定 的 分 组 聚集 值 ), 并 不 表示 你 需要 以 相同 的 方式 排序 输出 。 
应 该 提供 明确 的 ORDER BY 子 句 ， 即 使 其 效果 等 同 于 GROUP BY 子 句 。 









































提示 : 不 要 忘记 ORDER BY 
一 般 在 使 用 GROUP BY 子 多 时 ， 应 该 也 给 出 ORDER BY +). iX £ 
证 数据 正确 排序 的 唯一 方法 。 千 万 不 要 仅 依赖 GROUP BY 排序 数据 。 














为 说 明 GROUP BY 和 ORDER BY 的 使 用 方法 ,来 看 一 个 例子 .下 面 的 SELECT 
语句 类 似 于 前 面 那 些 例 子 。 它 检索 包含 三 个 或 更 多 物品 的 订单 号 和 订购 
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物品 的 数目 : 

输入 

SELECT order_num, COUNT(*) AS items 
FROM OrderItems 


GROUP BY order_num 
HAVING COUNT(*) >= 3; 


输出 


order_num items 


要 按 订 购物 品 的 数目 排序 输出 ， 需 要 添加 ORDER BY 子 句 ， 如 下 所 示 : 


输入 了 


SELECT order_num, COUNT(*) AS items 
FROM OrderItems 

GROUP BY order_num 

HAVING COUNT(*) >= 3 

ORDER BY items, order_num; 





说 明 : Access 的 不 兼容 性 

Microsoft Access 不 允许 按 别 名 排序 ， 因 此 这 个 例子 在 Access 中 将 失败 。 
解决 方法 是 用 实际 的 计算 或 字段 位 置 替 换 items( 在 ORDER BY 子 句 中 )， 
BP ORDER BY COUNT(*), order_num 或 ORDER BY 2, order_num, 








输出 了 
order_num items 
20006 3 
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20009 3 
20007 5 
20008 5 
3 Y 





在 这 个 例子 中 ， 使 用 GROUP BY 子 句 按 订单 号 (order_num 列 ) 分 组 数 
据 ， 以 便 COUNT (*) 函数 能 够 返回 每 个 订单 中 的 物品 数目 。HAVING FAJ 
过 滤 数 据 ， 使 得 只 返回 包含 三 个 或 更 多 物品 的 订单 。 最 后 ， 用 ORDER BY 
子 句 排序 输出 。 











10.5 SELECT 子 句 顺序 


下 面 回 顾 一 下 SELECT 语句 中 子 句 的 顺序 。 表 10-2 以 在 SELECT 语句 中 
使 用 时 必须 遵循 的 次 序 ， 列 出 迄今 为 止 所 学 过 的 子 句 。 





表 10-2 ”SELECT 子 句 及 其 顺序 
































子 “各 说 — BB 是 否 必须 使 用 
SELECT 要 返回 的 列 或 表达 式 是 

FROM 从 中 检索 数据 的 表 仅 在 从 表 选 择 数据 时 使 用 
WHERE 行 级 过 滤 T 

GROUP BY 分 组 说 明 仅 在 按 组 计算 聚集 时 使 用 
HAVING 组 级 过 滤 € 

ORDER BY 输出 排序 顺序 @ 





10.6 小结 








上 一 课 介绍 了 如 何 用 SQL 聚集 函数 对 数据 进行 汇总 计算 。 这 一 课 讲 授 了 
如 何 使 用 GROUP BY 子 句 对 多 组 数据 进行 汇总 计算 ， 返 回 每 个 组 的 结果 。 
我 们 看 到 了 如 何 使 用 HAVING 子 句 过 滤 特 定 的 组 ， 还 知道 了 ORDER BY 
和 GROUP BY 之 间 以 及 WHERE 和 HAVING 之 间 的 差异 。 
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这 一 课 介绍 什么 是 子 查询 ， 如 何 使 用 它们 。 


11.1 FEH 


SELECT 语句 是 SQL 的 查询 。 我 们 迄今 为 止 所 看 到 的 所 有 SELECT 语句 都 
是 简单 查询 ， 即 从 单个 数据 库 表 中 检索 数据 的 单条 语句 。 














查询 ( query ) 
任何 SQL 语句 都 是 查询 。 但 此 术语 一 般 指 SELECT 语句 。 














SQL 还 允许 创建 子 查 询 ( subquery )， 即 媒 套 在 其 他 查询 中 的 查询 。 为 什 
么 要 这 样 做 呢 ? 理解 这 个 概念 的 最 好 方法 是 考察 几 个 例子 。 




















说 明 : MySQL 支持 
如 果 使 用 MySQL， 应 该 知道 对 子 查询 的 支持 是 从 4.1 版 本 引入 的 。 
MySQL 的 早期 版 本 不 支持 子 查询 。 











11.2 利用 子 查询 进行 过 渡 


本 书 所 有 课 中 使 用 的 数据 库 表 都 是 关系 表 ( 关于 每 个 表 及 关系 的 描述 ， 
请 参阅 附录 A ) 订单 存储 在 两 个 表 中 ,每 个 订单 包含 订单 编号 、 客 户 ID. 
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订单 日 期 ,在 Orders 表 中 存储 为 一 行 。 各 订单 的 物品 存储 在 相关 的 
OrderItems R}, Orders 表 不 存储 顾客 信息 ， 只 存储 顾客 ID 。 顾 客 的 
实际 信息 存储 在 Customers 表 中 。 


现在 ， 假 如 需要 列 出 订购 物品 RGAN01 的 所 有 顾客 ， 应 该 怎样 检索 ? 下 
面 列 出 具体 的 步骤 。 








(1) 检索 包含 物品 RGAN01 的 所 有 订单 的 编号 。 
D 检索 具有 前 一 步骤 列 出 的 订单 编号 的 所 有 顾客 的 ID。 
(3) 检索 前 一 步骤 返回 的 所 有 顾客 ID 的 顾客 信息 。 

















上 述 每 个 步骤 都 可 以 单独 作为 一 个 查询 来 执行 。 可 以 把 一 条 SELECT 语 
名 返回 的 结果 用 于 另 一 条 SELECT 语句 的 WHERE 子 句 。 














岂可 以 使 用 子 查询 来 把 3 个 查询 组 合成 一 条 语句 。 








第 一 条 SELECT 语句 的 含义 很 明确 ， 它 对 prod_id 为 RGAN01 的 所 有 订 
单 物品 ， 检 索 其 order_num 列 。 输 出 列 出 了 两 个 包含 此 物品 的 订单 : 
输入 

SELECT order_num 


FROM OrderItems 
WHERE prod_id = 'RGANO1'; 


输出 


order_num 


现在 ,我 们 知道 了 哪个 订单 包含 要 检索 的 物品 , 下 一 步 查 询 与 订单 20007 
和 20008 相关 的 顾客 ID 。 利 用 第 5 课 介绍 的 IN T-HJ ,编写 如 下 的 SELECT 
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语句 : 


输入 了 


SELECT cust_id 
FROM Orders 
WHERE order_num IN (20007,20008); 


输出 了 


Cust_id 
1000000004 
1000000005 


























.2 利 
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现在 ,结合 这 两 个 查询 ， 把 第 一 个 查询 ( 返回 订单 号 的 那 一 个 ) 变 为 子 











查询 。 请 看 下 面 的 SELECT 语句 : 


输入 了 


SELECT cust_id 

FROM Orders 

WHERE order_num IN (SELECT order_num 
FROM OrderItems 


WHERE prod_id = 'RGANO1'); 


1000000004 
1000000005 


SY 
在 SELECT 语句 中 , 子 查询 总 是 从 内 疝 外 处 理 


Eo TEARI 





E EKI SELECT 语 








AJET, DBMS 实际 上 执行 了 两 个 操作 。 
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首先 ， 它 执行 下 面 的 查询 : 


SELECT order_num FROM orderitems WHERE prod_id='RGAN01' 
此 查询 返回 两 个 订单 号 : 20007 和 20008。 然后 , 这 两 个 值 以 IN 操作 
符 要 求 的 逗号 分 隔 的 格式 传递 给 外 部 查询 的 WHERE 子 句 。 外 部 查询 


变 成 : 








SELECT cust_id FROM orders WHERE order_num IN (20007,20008) 
可 以 看 到 ， 输 出 是 正确 的 ， 与 前 面 硬 编码 WHERE 子 句 所 返回 的 值 相同 。 























提示 : 格式 化 SOL 
包含 子 查 询 的 SELECT 语句 难以 阅读 和 调试 ， 它 们 在 较为 复杂 时 更 是 
如 此 。 如 上 所 示 ， 把 子 查询 分 解 为 多 行 并 进行 适当 的 缩 进 ， 能 极 大 地 
简化 子 查询 的 使 用 。 


顺便 一 提 ， 这 就 是 颜色 编码 起 作用 的 地 方 ， 好 的 DBMS 客户 端正 是 出 
于 这 个 原因 使 用 了 颜色 代码 SQL。 











现在 得 到 了 订购 物品 RGAN01 的 所 有 顾客 的 ID 。 下 一 步 是 检索 这 些 顾客 
ID 的 顾客 信息 。 检 索 两 列 的 SQL 语句 为 : 





输入 了 


SELECT cust_name, cust_contact 
FROM Customers 
WHERE cust_id IN ('1000000004','1000000005'); 











可 以 把 其 中 的 WHERE 子 句 转换 为 子 查询 ， 而 不 是 硬 编码 这 些 顾 客 ID : 


输入 了 


SELECT cust_name, cust_contact 
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FROM Customers 
WHERE cust_id IN (SELECT cust_id 
FROM Orders 
WHERE order_num IN (SELECT order_num 
FROM Orderltems 
WHERE prod_id = 'RGAN01')); 


输出 了 

cust_name cust_contact 

Ful í í í í 和 
The Toy Store Kim Howard 

SH Y 


为 了 执行 上 述 SELECT 语句 ，DBMS 实际 上 必须 执行 三 条 SELECT 语句 。 
最 里 边 的 子 查询 返回 订单 号 列表 ， 此 列表 用 于 其 外 面 的 子 查询 的 WHERE 
子 句 。 外 面 的 子 查询 返回 顾客 ID 列表 ， 此 顾客 ID 列表 用 于 最 外 层 查 询 
的 WHERE 子 句 。 最 外 层 查 询 返 回 所 需 的 数据 。 























可 见 , 在 WHERE 子 句 中 使 用 子 查 询 能 够 编写 出 功能 很 强 且 很 灵活 的 SQL 
语句 。 对 于 能 悉 套 的 子 查 询 的 数目 没有 限制 ， 不 过 在 实际 使 用 时 由 于 性 
能 的 限制 ,不 能 租 套 太 多 的 子 查 询 。 














注意 : 只 能 是 单列 
作为 子 查 询 的 SELECT 语句 只 能 查询 单个 列 。 企 图 检索 多 个 列 将 返回 


错误 。 








注意 : 子 查询 和 性 能 

这 里 给 出 的 代码 有 效 ， 并 且 获 得 了 所 需 的 结果 。 但 是 ， 使 用 子 查询 并 
不 总 是 执行 这 类 数据 检索 的 最 有 效 方法 。 更 多 的 论述 ， 请 参阅 第 12 
课 ， 其 中 将 再 次 给 出 这 个 例子 。 
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11.3 ”作为 计算 字段 使 用 子 查询 
使 用 子 查询 的 另 一 方法 是 创建 计算 字段 。 假 如 需要 显示 Customers 表 中 
每 个 顾客 的 订单 总 数 。 订 单 与 相应 的 顾客 ID 存储 在 Orders 表 中 。 








执行 这 个 操作 ， 要 遵循 下 面 的 步 又 : 


(1) 从 Customers 表 中 检索 顾客 列表 ; 
(2) 对 于 检索 出 的 每 个 顾客 ,统计 其 在 Orders 表 中 的 订单 数目 。 





正如 前 两 课 所 述 ， 可 以 使 用 SELECT COUNT(* 对 表 中 的 行进 行 计 数 ， 并 
且 通 过 提供 一 条 WHERE 子 句 来 过 滤 某 个 特定 的 顾客 ID ， 仅 对 该 顾客 的 订 
单 进 行 计 数 。 例 如 ， 下 面 的 代码 对 顾客 1000000001 的 订单 进行 计数 : 























输入 了 


SELECT COUNT(*) AS orders 
FROM Orders 
WHERE cust_id = '1000000001'; 








要 对 每 个 顾客 执行 COUNT G) , 应 该 将 它 作为 一 个 子 查 询 。 请 看 下 面 的 代码 : 


输入 了 


SELECT cust_name， 
cust_state, 
(SELECT COUNT(*) 
FROM Orders 
WHERE Orders.cust_id = Customers.cust_id) AS orders 
FROM Customers 
ORDER BY cust_name; 


输出 了 


CUSt_name cust_state orders 
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Fun4A11 IN 1 
Fun4A11 AZ 1 
Kids Place OH 0 
The Toy Store IL 1 
Village Toys MI 2 
Biy 


这 条 SELECT 语句 对 Customers 表 中 每 个 顾客 返回 三 列 : cust name, 
cust_state 和 orders。orders 是 一 个 计算 字段 ， 它 是 由 圆 括号 中 的 
子 查询 建立 的 。 该 子 查 询 对 检索 出 的 每 个 顾客 执行 一 次 。 在 此 例 中 ， 该 
子 查询 执行 了 5 次 ， 因 为 检索 出 了 5 个 顾客 。 




















子 查询 中 的 WHERE 子 句 与 前 面 使 用 的 WHERE 子 句 稍 有 不 同 , 因为 它 使 用 了 
完全 限定 列 名 ， 而 不 只 是 列 名 (cust_id )。 它 指定 表 名 和 列 名 
(Orders.cust_id 和 Customers .cust id) 下 面 的 WHERE 子 句 告诉 SQL, 
比较 0rders 表 中 的 cust_id 和 当前 正 从 Customers 表 中 检索 的 cust_id: 


























WHERE Orders.cust_id = Customers .cust_id 

用 一 个 句点 分 隔 表 名 和 列 名 ， 在 有 可 能 混淆 列 名 时 必须 使 用 这 种 语法 。 
在 这 个 例子 中 ， 有 两 个 cust_id 列 : 一 个 在 Customers 中 ， 男 一 个 在 
Orders 中 。 如 果 不 采用 完全 限定 列 名 ,DBMS 会 认为 要 对 Orders KP 
的 cust_id 自身 进行 比较 。 因 为 




















SELECT COUNT(*) FROM Orders WHERE cust_id = cust_id 
总 是 返回 Orders 表 中 订单 的 总 数 ， 而 这 个 结果 不 是 我 们 想 要 的 : 

















输入 了 


SELECT cust_name， 
cust_state, 
(SELECT COUNT(*) 

FROM Orders 
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WHERE cust_id = cust_id) AS orders 
FROM Customers 
ORDER BY cust_name ; 


输出 了 

cust_name cust_state orders 
Fun4A11 IN 5 
Fun4A11 AZ 5 

Kids Place OH 5 

The Toy Store IL 5 
Village Toys MI 5 





虽然 子 查 询 在 构造 这 种 SELECT 语句 时 极 有 用 ,但 必须 注意 限制 有 此 义 的 列 。 








注意 : 完全 限定 列 名 

你 已 经 看 到 了 为 什么 要 使 用 完全 限定 列 名 ,没有 具体 指定 就 会 返回 错 
误 结果 ， 因 为 DBMS 会 误解 你 的 意思 。 有 了 时候， 由 于 出 现 冲 突 列 名 而 
导致 的 歧义 性 , 会 引起 DBMS 抛 出 错误 信息 。 例 如 ，WHERE 或 ORDER 
BY 子 句 指定 的 菜 个 列 名 可 能 会 出 现在 多 个 表 中 。 好 的 做 法 是 ， 如 果 在 
SELECT 语句 中 操作 多 个 表 ， 就 应 使 用 完全 限定 列 名 来 避免 歧义 。 








提示 : 不 止 一 种 解决 方案 

正如 这 一 课 前 面 所 述 ， 虽 然 这 里 给 出 的 样 例 代码 运行 良好 ， 但 它 并 不 
是 解决 这 种 数据 检索 的 最 有 效 方法 。 在 后 面 两 课 学 习 JOIN ñ, 我们 
还 会 遇 到 这 个 例子 。 








11.4 小结 


这 一 课 学 习 了 什么 是 子 查 询 ， 如 何 使 用 它们 。 子 查询 常用 于 WHERE FAJ 
的 IN 操作 符 中 ,以 及 用 来 填充 计算 列 。 我 们 举 了 这 两 种 操作 类 型 的 例子 。 
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第 12 课 联结 表 

















这 一 课 会 介绍 什么 是 联结 ， 为 什么 使 用 联结 ， 如 何 编写 使 用 联结 的 
SELECT 语句 。 








12.1 联结 


SQL 最 强大 的 功能 之 一 就 是 能 在 数据 查询 的 执行 中 联结 (join ) 表 。 联 结 
是 利用 SQL 的 SELECT 能 执行 的 最 重要 的 操作 ， 很 好 地 理解 联结 及 其 语 
法 是 学 习 SQL 的 极为 重要 的 部 分 。 

















在 能 够 有 效 地 使 用 联结 前 ， 必 须 了 解 关 系 表 以 及 关系 数据 库 设 计 的 一 些 
基础 知识 。 下 面 的 介绍 并 不 能 涵盖 这 一 主题 的 所 有 内 容 ， 但 作为 入 门 已 
AET. 





12.1.1 关系 表 
理解 关系 表 ， 最 好 是 来 看 个 例子 。 

















有 一 个 包含 产品 目录 的 数据 库 表 ， 其 中 每 类 物品 占 一 行 。 对 于 每 一 种 物 
品 ， 要 存储 的 信息 包括 产品 描述 、 价 格 ， 以 及 生产 该 产品 的 供应 商 。 








现在 有 同一 供应 商 生 产 的 多 种 物品 ， 那 么 在 何 处 存储 供应 商 名 、 地 址 、 
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联系 方法 等 供应 商 信 息 呢 ? 将 这 些 数据 与 产品 信息 分 开 存储 的 理由 是 : 





口 同一 供应 商 生产 的 每 个 产品 ， 其 供应 商 信息 都 是 相同 的 ， 对 每 个 产品 
重复 此 信息 既 浪费 时 间 又 浪费 存储 空间 ; 
口 如 果 供 应 商 信息 发 生变 化 ， 例 如 供应 商 迁 址 或 电话 号 码 变动 ， 只 需 
改 一 次 即 可 ; 

口 如 果 有 重复 数据 ( 即 每 种 产品 都 存储 供应 商 信息 )， 则 很 难保 证 每 次 
输入 该 数据 的 方式 都 相同 。 不 一 致 的 数据 在 报表 中 就 很 难 利 用 。 

















< 














关键 是 ， 相 同 的 数据 出 现 多 次 决 不 是 一 件 好事 ， 这 是 关系 数据 库 设 计 的 
基础 。 关 系 表 的 设计 就 是 要 把 信息 分 解 成 多 个 表 ， 一 类 数据 一 个 表 。 各 
表 通 过 某 些 共同 的 值 互相 关联 (所 以 才 叫 关系 数据 库 )。 





5 























在 这 个 例子 中 可 建立 两 个 表 : 一 个 存储 供应 商 信息 ， 另 一 个 存储 产品 信 
Ho Vendors 表 包 含 所 有 供应 商 信息 ， 每 个 供应 商 占 一 行 ， 具 有 唯一 的 
标识 。 此 标识 称 为 主键 (primary key )， 可 以 是 供应 商 ID 或 任何 其 他 唯 
一 值 。 




















Products 表 只 存储 产品 信息 , 除了 存储 供应 商 ID (Vendors 表 的 主键 ) 
外 ， 它 不 存储 其 他 有 关 供 应 商 的 信息 。Vendors 表 的 主键 将 Vendors R 
与 Products 表 关 联 ， 利 用 供应 商 ID 能 从 Vendors 表 中 找 出 相应 供应 
商 的 详细 信息 。 








这 样 做 的 好 处 是 : 








口 供应 商 信息 不 重复 ， 不 会 浪费 时 间 和 空间 ; 

口 如 果 供 应 商 信息 变动 , 可 以 只 更 新 Vendors 表 中 的 单个 记录 , 相关 表 
中 的 数据 不 用 改动 ; 

口 由 于 数据 不 重复 ,数据 显然 是 一 致 的 ,使 得 处 理 数据 和 生成 报表 更 简单 。 
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总 之 ， 关 系数 据 可 以 有 效 地 存储 ， 方 便 地 处 理 。 因 此 ， 关 系数 据 库 的 可 
伸缩 性 远 比 非 关 系数 据 库 要 好 。 








可 伸缩 ( scale ) 
能 够 适应 不 断 增 加 的 工作 量 而 不 失败 。 设 计 良 好 的 数据 库 或 应 用 程序 
称 为 可 伸缩 性 好 ( scale well )。 











12.1.2 ”为 什么 使 用 联结 


如 前 所 述 ， 将 数据 分 解 为 多 个 表 能 更 有 效 地 存储 ， 更 方便 地 处 理 ， 并 且 
可 伸缩 性 更 好 。 但 这 些 好 处 是 有 代价 的 。 























如 果 数 据 存储 在 多 个 表 中 ， 怎 样 用 一 条 SELECT 语句 就 检索 出 数据 呢 ? 
































答案 是 使 用 联结 。 简 单 说 ， 联 结 是 一 种 机 制 ， 用 来 在 一 条 SELECT 语句 
中 关联 表 ， 因 此 称 为 联结 。 使 用 特殊 的 语法 ， 可 以 联结 多 个 表 返 回 一 组 
输出 ， 联 结 在 运行 时 关联 表 中 正确 的 行 。 














说 明 : 使 用 交互 式 DBMS 工具 
重要 的 是 ， 要 理解 联结 不 是 物理 实体 。 换 多 话说 ， 它 在 实际 的 数据 库 表 
中 并 不 存在 , DBMS 会 根据 需要 建立 联结 , 它 在 查询 执行 期 间 一 直 存 在 。 


许多 DBMS 提供 图 形 界面 ， 用 来 交互 式 地 定义 表 关 系 。 这 些 工具 极其 
有 助 于 维护 引用 完整 性 。 在 使 用 关系 表 时 ， 仅 在 关系 列 中 插入 合法 数 
据 是 非常 重要 的 。 回 到 这 里 的 例子 ， 如 果 Products 表 中 存储 了 无 效 
的 供应 商 ID ， 则 相应 的 产品 不 可 访问 ， 因 为 它们 没有 关联 到 某 个 供应 
商 。 为 避免 这 种 情况 发 生 ， 可 指示 数据 库 只 允许 在 Products 表 的 供 
应 商 ID 列 中 出 现 合法 值 ( 即 出 现在 Vendors 表 中 的 供应 商 )。 引用 完 
整 性 表示 DBMS 强制 实施 数据 完整 性 规则 。 这 些 规则 一 般 由 提供 了 界 
面 的 DBMS 管理 。 
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12.2 创建 联结 

创建 联结 非常 简单 ， 指 定 要 联结 的 所 有 表 以 及 关联 它们 的 方式 即 可 。 请 
看 下 面 的 例子 : 


输入 了 


SELECT vend_name，prod_name，prod_price 
FROM Vendors，Products 
WHERE Vendors.vend_id = Products.vend_id; 


输出 了 

vend_name prod_name prod_price 
Doll House Inc. Fish bean bag toy 3.4900 
Doll House Inc. Bird bean bag toy 3.4900 
Doll House Inc. Rabbit bean bag toy 3.4900 
Bears R Us 8 inch teddy bear 5.9900 
Bears R Us 12 inch teddy bear 8.9900 
Bears R Us 18 inch teddy bear 11.9900 
Doll House Inc. Raggedy Ann 4.9900 
Fun and Games King doll 9.4900 
Fun and Games Queen doll 9.4900 
5 Y 


我 们 来 看 这 段 代码 。SELECT 语句 与 前 面 所 有 语句 一 样 指定 要 检索 的 列 。 
这 里 最 大 的 差别 是 所 指定 的 两 列 (prod_name 和 prod_price ) 在 一 个 
表 中 ， 而 第 三 列 ( vend_name ) 在 另 一 个 表 中 。 








现在 来 看 FROM 子 句 。 与 以 前 的 SELECT 语句 不 一 样 ， 这 条 语句 的 FROM 
子 句 列 出 了 两 个 表 : Vendors 和 Products。 它 们 就 是 这 条 SELECT 语句 
联结 的 两 个 表 的 名 字 。 这 两 个 表 用 WHERE 子 句 正确 地 联结 ，WHERE FA 
指示 DBMS 将 Vendors 表 中 的 vend_id 与 Products 表 中 的 vend_id PE 
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配 起 来 。 








可 以 看 到 ,要 匹配 的 两 列 指定 为 Vendors.vend_id 和 Products.vend_id。 
这 里 需要 这 种 完全 限定 列 名 ， 如 果 只 给 出 vend_id, DBMS 就 不 知道 指 的 
是 哪 一 个 (每 个 表 中 有 一 个 )。 从 前 面 的 输出 可 以 看 到 , 一 条 SELECT 语句 
返回 了 两 个 不 同 表 中 的 数据 。 








警告 : 完全 限定 列 名 

就 像 前 一 课 提 到 的 ， 在 引用 的 列 可 能 出 现 歧义 时 ， 必 须 使 用 完全 限定 
列 名 ( 用 一 个 句点 分 隔 表 名 和 列 名 )。 如 果 引 用 一 个 没有 用 表 名 限制 的 
具有 歧义 的 列 名 ， 大 多 数 DBMS 会 返回 错误 。 











12.2.1 WHERE 子 句 的 重要 性 


使 用 WHERE 子 句 建立 联结 关系 似乎 有 点 奇怪 ,但 实际 上 是 有 个 很 充分 的 
理由 的 。 要 记 住 ， 在 一 条 SELECT 语句 中 联结 几 个 表 时 ， 相 应 的 关系 是 
在 运行 中 构造 的 。 在 数据 库 表 的 定义 中 没有 指示 DBMS 如 何 对 表 进 行 联 
结 的 内 容 。 你 必须 自己 做 这 件 事 情 。 在 联结 两 个 表 时 ， 实 际 要 做 的 是 将 
第 一 个 表 中 的 每 一 行 与 第 二 个 表 中 的 每 一 行 配对 。WHERE 子 句 作为 过 滤 
条 件 ， 只 包含 那些 匹配 给 定 条 件 ( 这 里 是 联结 条 件 ) 的 行 。 没 有 WHERE 
子 句 ， 第 一 个 表 中 的 每 一 行将 与 第 二 个 表 中 的 每 一 行 配对 ， 而 不 管 它们 
逻辑 上 是 否 能 配 在 一 起 。 















































向 卡 儿 积 ( cartesian product ) 
由 没有 联结 条 件 的 表 关 系 返 回 的 结果 为 笛 卡 儿 积 。 检 索 出 的 行 的 数目 
将 是 第 一 个 表 中 的 行 数 乘 以 第 二 个 表 中 的 行 数 。 











理解 这 一 点 ， 请 看 下 面 的 SELECT 语句 及 其 输出 : 
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输入 了 





联结 表 


SELECT vend_name，prod_name，prod_price 
FROM Vendors, Products; 


prod_name 


输出 了 

vend_name 
Bears R Us 
Bears R Us 
Bears R Us 
Bears R Us 
Bears R Us 
Bears R Us 
Bears R Us 
Bears R Us 
Bears R Us 


Bear Emporium 
Bear Emporium 
Bear Emporium 
Bear Emporium 
Bear Emporium 
Bear Emporium 
Bear Emporium 
Bear Emporium 
Bear Emporium 
Doll House 
Doll House 
Doll House 
Doll House 
Doll House 
Doll House 
Doll House 
Doll House 
Doll House 
Inc. 
Inc. 
Inc. 
Inc. 
Inc. 


Furball 
Furball 
Furball 
Furball 
Furball 


Inc. 
Inc. 
Inc. 
Inc. 
Inc. 
Inc. 
Inc. 
Inc. 
Inc. 


8 inch teddy bear 
12 inch teddy bear 
18 inch teddy bear 
Fish bean bag toy 
Bird bean bag toy 
Rabbit bean bag toy 
Raggedy Ann 

King doll 

Queen doll 

8 inch teddy bear 
12 inch teddy bear 
18 inch teddy bear 
Fish bean bag toy 
Bird bean bag toy 
Rabbit bean bag toy 
Raggedy Ann 

King doll 

Queen doll 

8 inch teddy bear 
12 inch teddy bear 
18 inch teddy bear 
Fish bean bag toy 
Bird bean bag toy 
Rabbit bean bag toy 
Raggedy Ann 

King doll 

Queen doll 

8 inch teddy bear 
12 inch teddy bear 
18 inch teddy bear 
Fish bean bag toy 
Bird bean bag toy 
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Furball Inc. Rabbit bean bag toy 3.49 
Furball Inc. Raggedy Ann 4.99 
Furball Inc. King doll 9.49 
Furball Inc. Queen doll 9.49 
Fun and Games 8 inch teddy bear 5.99 
Fun and Games 12 inch teddy bear 8.99 
Fun and Games 18 inch teddy bear 11.99 
Fun and Games Fish bean bag toy 3.49 
Fun and Games Bird bean bag toy 3.49 
Fun and Games Rabbit bean bag toy 3.49 
Fun and Games Raggedy Ann 4.99 
Fun and Games King doll 9.49 
Fun and Games Queen doll 9.49 
Jouets et ours 8 inch teddy bear 5.99 
Jouets et ours 12 inch teddy bear 8.99 
Jouets et ours 18 inch teddy bear 11.99 
Jouets et ours Fish bean bag toy 3.49 
Jouets et ours Bird bean bag toy 3.49 
Jouets et ours Rabbit bean bag toy 3.49 
Jouets et ours Raggedy Ann 4.99 
Jouets et ours King doll 9.49 
Jouets et ours Queen doll 9.49 
SH Y 

















从 上 面 的 输出 可 以 看 到 ， 相 应 的 笛 卡 儿 积 不 是 我 们 想 要 的 。 这 里 返回 的 
数据 用 每 个 供应 商 匹 配 了 每 个 产品 ， 包 括 了 供应 商 不 正确 的 产品 (即使 
供应 商 根本 就 没有 产品 )。 

















PR 





注意 : 不 要 忘 了 WHERE 子 句 

要 保证 所 有 联结 都 有 WHERE 子 句 ,否则 DBMS 将 返回 比 想 要 的 数据 多 
得 多 的 数据 。 同 理 ， 要 保证 WHERE 子 名 的 正确 性 。 不 正确 的 过 滤 条 件 
会 导致 DBMS 返回 不 正确 的 数据 。 








提示 : 叉 联结 
有 了 时， 返回 箭 卡 儿 积 的 联结 ， 也 称 又 联结 〈cross join )。 
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12.2.2 ”内 联结 

目前 为 止 使 用 的 联结 称 为 等 值 联 结 ( equijoin )， 它 基于 两 个 表 之 间 的 相 
等 测试 。 这 种 联结 也 称 为 内 联结 (innerjoin )。 其 实 ， 可 以 对 这 种 联结 
用 稍微 不 同 的 语法 ， 明 确 指定 联结 的 类 型 。 下 面 的 SELECT 语句 返回 与 
前 面 例子 完全 相同 的 数据 : 








输入 了 


SELECT vend_name，prod_name，prod_price 
FROM Vendors INNER JOIN Products 
ON Vendors.vend_id = Products.vend_id; 


BY 


此 语句 中 的 SELECT 与 前 面 的 SELECT 语句 相同 ,但 FROM 子 句 不 同 。 这 
里 ,两 个 表 之 间 的 关系 是 以 INNER JOIN 指定 的 部 分 FROM 子 句 。 在 使 用 
这 种 语法 时 ， 联 结 条 件 用 特定 的 ON 子 句 而 不 是 WHERE 子 句 给 出 。 传 递 
给 ON 的 实际 条 件 与 传递 给 WHERE 的 相同 。 








至 于 选用 哪 种 语法 ， 请 参阅 具体 的 DBMS 文档 。 





说 明 :“ 正 确 的 ”语法 

ANSI SQL 规范 首选 INNER JOIN 语法 ， 之 前 使 用 的 是 简单 的 等 值 语 
法 。 其 实 ，SQL 语言 纯正 论 者 是 用 部 视 的 眼光 看 待 简单 语法 的 。 这 就 
是 说 ，DBMS 的 确 支持 简单 格式 和 标准 格式 ,我 建议 你 要 理解 这 两 种 
格式 ， 有 具体 使 用 就 看 你 用 哪个 更 顺手 了 。 











12.2.3 ”联结 多 个 表 


SQL 不 限制 一 条 SELECT 语句 中 可 以 联结 的 表 的 数目 。 创 建 联结 的 基本 
规则 也 相同 。 首 先 列 出 所 有 表 ， 然 后 定义 表 之 间 的 关系 。 例 如 : 
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输入 了 


SELECT prod_name, vend_name, prod_price, quantity 
FROM OrderItems, Products, Vendors 

WHERE Products.vend_id = Vendors.vend_id 

AND OrderItems.prod_id = Products.prod_id 

AND order_num = 20007; 


输出 了 

prod_name vend_name prod_price quantity 
18 inch teddy bear Bears R Us 11.9900 50 

Fish bean bag toy Doll House Inc. 3.4900 100 

Bird bean bag toy Doll House Inc. 3.4900 100 
Rabbit bean bag toy Doll House Inc. 3.4900 100 
Raggedy Ann Doll House Inc. 4.9900 50 
SH Y 


这 个 例子 显示 订单 20007 中 的 物品 ,订单 物品 存储 在 OrderItems 表 中 。 
每 个 产品 按 其 产品 ID 存储 , 它 引 用 Products 表 中 的 产品 。 这 些 产品 通 
过 供应 商 ID 联结 到 Vendors 表 中 相应 的 供应 商 , 供应 商 ID 存储 在 每 个 
产品 的 记录 中 。 这 里 的 FROM 子 句 列 出 三 个 表 , WHERE 子 句 定义 这 两 个 联 
结 条 件 ， 而 第 三 个 联结 条 件 用 来 过 滤 出 订单 20007 中 的 物品 。 














注意 : 性 能 考虑 

DBMS 在 运行 时 关联 指定 的 每 个 表 ， 以 处 理 联结 。 这 种 处 理 可 能 非常 
耗费 资源 ， 因 此 应 该 注意 ， 不 要 联结 不 必要 的 表 。 联 结 的 表 越 多 ， 性 
能 下 降 越 厉害 。 








注意 : 联结 中 表 的 最 大 数目 
虽然 SQL 本身 不 限制 每 个 联结 约束 中 表 的 数目 , 但 实际 上 许多 DBMS 
都 有 限制 。 请 参阅 具体 的 DBMS 文档 以 了 解 其 限制 。 
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现在 回顾 一 下 第 11 课 中 的 例子 ， 如 下 的 SELECT 语句 返回 订购 产品 
RGAN01 的 顾客 列表 : 
输入 了 


SELECT cust_name, cust_contact 
FROM Customers 
WHERE cust_id IN (SELECT cust_id 
FROM Orders 
WHERE order_num IN (SELECT order_num 
FROM OrderItems 
WHERE prod_id = 'RGAN01')); 


如 第 11 课 所 述 ， 子 查询 并 不 总 是 执行 复杂 SELECT 操作 的 最 有 效 方法 ， 
下 面 是 使 用 联结 的 相同 查询 : 


输入 了 


SELECT cust_name, cust_contact 

FROM Customers, Orders, OrderItems 

WHERE Customers.cust_id = Orders.cust_id 

AND OrderItems.order_num = Orders.order_num 
AND prod_id = 'RGANO1'; 

















输出 了 

cust_name cust_contact 

Fun | Denise ee 
The Toy Store Kim Howard 

S Y 


如 第 11 课 所 述 ， 这 个 查询 中 的 返回 数据 需要 使 用 3 个 表 。 但 在 这 里 ,我 们 
没有 在 般 套 子 查 询 中 使 用 它们 ， 而 是 使 用 了 两 个 联结 来 连接 表 。 这 里 有 三 
个 WHERE 子 句 条 件 。 前 两 个 关联 联结 中 的 表 ， 后 一 个 过 滤 产 品 RGAN01 的 
数据 。 
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提示 : 多 做 实验 

可 以 看 到 ,执行 任 一 给 定 的 | 操作 一 般 不 止 一 种 方法 。 很 少 有 绝对 
正确 或 绝对 错误 的 方法 。 性 能 可 能 会 受 操作 类 型 、 所 使 用 的 DBMS、 
表 中 数据 量 、 是 否 存 在 索引 响 。 因 此 ， 有 必要 试验 不 
同 的 选择 机 制 ， 找 出 最 适合 具体 情况 的 方法 。 








12.3 ”小结 

联结 是 SQL 中 一 个 最 重要 、 最 强大 的 特性 ， 有 效 地 使 用 联结 需要 对 关系 
数据 库 设计 有 基本 的 了 解 。 本 课 在 介绍 联结 时 ， 讲 述 了 一 些 关系 数据 库 
设计 的 基本 知识 ， 包 括 等 值 联结 〈 也 称 为 内 联结 ) 这 种 最 常用 的 联结 。 
下 一 课 将 介绍 如 何 创建 其 他 类 型 的 联结 。 
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本 课 讲 解 男 外 一 些 联结 ( 包括 它们 的 含义 和 使 用 方法 ) 介绍 如 何 使 用 表 
别名 ， 如 何 对 被 联结 的 表 使 用 聚集 函数 。 





13.1 使 用 表 别 名 
第 7 课 介绍 了 如 何 使 用 别名 引用 被 检索 的 表 列 。 给 列 起 别名 的 语法 如 下 : 





输入 了 


SELECT RTRIM(vend_name) + ' (' + RTRIM(vend_country) + ')' 
AS vend_title 

FROM Vendors 

ORDER BY vend_name; 


SQL 除了 可 以 对 列 名 和 计算 字段 使 用 别名 ， 还 允许 给 表 名 起 别名 。 这 样 
做 有 两 个 主要 理由 : 


口 缩短 SQL 语句 ; 
口 允许 在 一 条 SELECT 语句 中 多 次 使 用 相同 的 表 。 





请 看 下 面 的 SELECT 语句 。 它 与 前 一 课 例子 中 所 用 的 语句 基本 相同 ， 但 
改 成 了 使 用 别名 : 
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输入 了 


SELECT cust_name, cust_contact 

FROM Customers AS C, Orders AS O, OrderItems AS OI 
WHERE C.cust_id = O.cust_id 

AND Ol.order_num = O.order_num 

AND prod_id = 'RGANO1'; 


BY 


可 以 看 到 ，FROM 子 句 中 的 三 个 表 全 都 有 别名 。Customers AS C 使 用 C 
作为 Customers 的 别名 ， 如 此 等 等 。 这 样 ， 就 可 以 使 用 省 略 的 C 而 不 用 
全 名 Customers。 在 这 个 例子 中 ， 表 别名 只 用 于 WHERE FAs EKER 
仅 能 用 于 WHERE 子 句 ,还 可 以 用 于 SELECT 的 列表 、ORDER BY 子 句 以 及 
其 他 语句 部 分 。 




















注意 : Oracle 中 没有 AS 
Oracle 不 支持 AS 关键 字 。 要 在 Oracle 中 使 用 别名 ， 可 以 不 用 As， 简单 
地 指定 列 名 即 可 ( 因此 , 应 该 是 Customers C, 而 不 是 Customers AS C), 














需要 注意 ， 表 别名 只 在 查询 执行 中 使 用 。 与 列 别名 不 一 样 ， 表 别名 不 返 
回 到 客户 端 。 
13.2 ”使 用 不 同类 型 的 联结 


迄今 为 止 , 我 们 使 用 的 只 是 内 联结 或 等 值 联结 的 简单 联结 。 现 在 来 看 三 种 
其 他 联结 : 自 联 结 ( self-join ). 自然 联结 (natural join ) 和 外 联结 ( outer join )。 


























13.2.1 AKA 
如 前 所 述 ， 使 用 表 别 名 的 一 个 主要 原因 是 能 在 一 条 SELECT 语句 中 不 止 
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一 次 引用 相同 的 表 。 下 面 举 一 个 例子 。 





假如 要 给 与 Jim Jones 同一 公司 的 所 有 顾客 发 送 一 封 信件 。 这 个 查询 要 求 
首先 找 出 Jim Jones 工作 的 公司 ， 然 后 找 出 在 该 公司 工作 的 顾客 。 下 面 是 
解决 此 问题 的 一 种 方法 : 


输入 了 


SELECT cust_id, cust_name, cust_contact 

FROM Customers 

WHERE cust_name = (SELECT cust_name 
FROM Customers 


WHERE cust_contact = 'Jim Jones'); 
输出 了 
cust_id cust_name cust_contact 
1000000003 Full Jim Jones 
1000000004 Fun4A11 Denise L. Stephens 
Biy 


这 是 第 一 种 解决 方案 ， 使 用 了 子 查询 。 内 部 的 SELECT 语句 做 了 一 个 简 
单 检 索 , 返回 Jim Jones 工作 公司 的 cust_name。 该 名 字 用 于 外 部 查询 的 
WHERE 子 句 中 ， 以 检索 出 为 该 公司 工作 的 所 有 雇员 (第 11 课 中 讲授 了 子 
查询 ， 更 多 信息 请 参阅 该 课 )。 











现在 来 看 使 用 联结 的 相同 查询 : 
输入 了 








SELECT cl.cust_id, cl.cust_name, cl.cust_contact 
FROM Customers AS cl, Customers AS c2 

WHERE cl.cust_name = c2.cust_name 

AND c2.cust_contact = 'Jim Jones'; 
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输出 了 

Cust_id Cust_name cust_contact 
1000000003 Fun4A11 Jim Jones 
1000000004 Fun4A11 Denise L. Stephens 





提示 : Oracle 中 没有 AS 
Oracle 用 户 应 该 记 住 去 掉 AS. 








Syri Y 

此 查询 中 需要 的 两 个 表 实际 上 是 相同 的 表 ， 因 此 Customers 表 在 FROM 
子 句 中 出 现 了 两 次 。 虽 然 这 是 完全 合法 的 , 但 对 Customers 的 引用 具有 
BHE, KH DBMS 不 知道 你 引用 的 是 哪个 Customers 表 。 


















































解决 此 问题 ， 需 要 使 用 表 别 名 。Customers 第 一 次 出 现 用 了 别名 C1， 第 
二 次 出 现 用 了 别名 C2。 现 在 可 以 将 这 些 别名 用 作 表 名 。 例 如 ，SELECT 语 
句 使 用 C1 前 绥 明 确 给 出 所 需 列 的 全 名 。 如 果 不 这 样 , DBMS 将 返回 错误 ， 
因为 名 为 cust_id、cust_name、cust_contact 的 列 各 有 两 个 。 DBMS 
不 知道 想 要 的 是 哪 一 列 ( 即使 它们 其 实 是 同一 列 )。WHERE 首先 联结 两 个 
表 ， 然 后 按 第 二 个 表 中 的 cust_contact 过 滤 数 据 ， 返 回 所 需 的 数据 。 























提示 : 用 自 联 结 而 不 用 子 查询 

自 联 结 通常 作为 外 部 语句 ， 用 来 替代 从 相同 表 中 检索 数据 的 使 用 子 查 
询 语 句 。 虽 然 最 终 的 结果 是 相同 的 , 但 许多 DBMS 处 理 联结 远 比 处 理 
子 查询 快 得 多 。 应 该 试 一 下 两 种 方法 ， 以 确定 哪 一 种 的 性 能 更 好 。 











13.2.2 ”自然 联结 
无 论 何 时 对 表 进行 联结 ， 应 该 至 少 有 一 列 不 止 出 现在 一 个 表 中 ( 被 联结 
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的 列 o 标准 的 联结 ( 前 一 课 中 介绍 的 内 联结 ) 返回 所 有 数据 ， 相 同 的 列 
甚至 多 次 出 现 。 自 然 联 结 排除 多 次 出 现 ， 使 每 一 列 只 返回 一 次 。 





怎样 完成 这 项 工作 呢 ? 答案 是 , 系统 不 完成 这 项 工作 , 由 你 自己 完成 它 。 
自然 联结 要 求 你 只 能 选择 那些 唯一 的 列 ， 一 般 通 过 对 一 个 表 使 用 通配符 
( SELECT * ), 而 对 其 他 表 的 列 使 用 明确 的 子 集 来 完成 。 下面 举 一 个 例子 : 











输入 了 


SELECT C.*, O.order_num, O.order_date, 
OI.prod_id, OI.quantity, OI.item_price 
FROM Customers AS C, Orders AS O, OrderItems AS OI 
WHERE C.cust_id = O.cust_id 
AND Ol.order_num = O.order_num 
AND prod_id = 'RGANO1'; 





提示 : Oracle 中 没有 AS 
Oracle 用 户 应 该 记 住 去 掉 AS. 











BY 


在 这 个 例子 中 ， 通 配 符 只 对 第 一 个 表 使 用 。 所 有 其 他 列 明确 列 出 ， 所 以 
没有 重复 的 列 被 检索 出 来 。 




















事实 上 ， 我 们 迄今 为 止 建立 的 每 个 内 联结 都 是 自然 联结 ， 很 可 能 永远 都 
不 会 用 到 不 是 自然 联结 的 内 联结 。 




















13.23 ”外 联结 


许多 联结 将 一 个 表 中 的 行 与 另 一 个 表 中 的 行 相关 联 ， 但 有 时 候 需 要 包含 
没有 关联 行 的 那些 行 。 例 如 ， 可 能 需要 使 用 联结 完成 以 下 工作 : 
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口 对 每 个 顾客 下 的 订单 进行 计数 ， 包 括 那 些 至 今 尚未 下 订单 的 顾客 ; 
口 列 出 所 有 产品 以 及 订购 数量 ， 包 括 没有 人 订购 的 产品 ; 
口 计算 平均 销售 规模 ， 包 括 那 些 至 今 尚 未 下 订单 的 顾客 。 








在 上 述 例子 中 ， 联 结 包 含 了 那些 在 相关 表 中 没有 关联 行 的 行 。 这 种 联结 
称 为 外 联结 。 





注意 : 语法 差别 

需要 注意 ， 用 来 创建 外 联结 的 语法 在 不 同 的 SQL 实现 中 可 能 稍 有 不 
同 。 下 面 段落 中 描述 的 各 种 语法 形式 覆盖 了 大 多 数 实现 ， 在 继续 学 习 
之 前 请 参阅 你 使 用 的 DBMS 文档 ， 以 确定 其 语法 。 














下 面 的 SELECT 语句 给 出 了 一 个 简单 的 内 联结 。 它 检索 所 有 顾客 及 其 订单 : 


输入 
SELECT Customers.cust_id, Orders.order_num 


FROM Customers INNER JOIN Orders 
ON Customers.cust_id = Orders.cust_id; 


外 联结 语法 类 似 。 要 检索 包括 没有 订单 顾客 在 内 的 所 有 顾客 , 可 如 下 进行 : 





输入 了 


SELECT Customers.cust_id, Orders.order_num 
FROM Customers LEFT OUTER JOIN Orders 
ON Customers.cust_id = Orders.cust_id; 


输出 


cust_id order_num 


1000000001 20005 
1000000001 20009 
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1000000002 NULL 
1000000003 20006 
1000000004 20007 
1000000005 20008 
BY 


类 似 上 一 课 提 到 的 内 联结 ， 这 条 SELECT 语句 使 用 了 关键 字 OUTER JOIN 
来 指定 联结 类 型 ( 而 不 是 在 WHERE 子 句 中 指定 ),。 但 是 ,与 内 联结 关联 
两 个 表 中 的 行 不 同 的 是 ， 外 联结 还 包括 没有 关联 行 的 行 。 在 使 用 OUTER 
JOIN 语法 时 ， 必 须 使 用 RIGHT 或 LEFT 关键 字 指定 包括 其 所 有 行 的 表 
( RIGHT 指出 的 是 OUTER JOIN 右边 的 表 ， 而 LEFT 指出 的 是 OUTER JOIN 
左边 的 表 )。 上 面 的 例子 使 用 LEFT OUTER JOIN 从 FROM 子 句 左边 的 表 
( Customers K ) 中 选择 所 有 行 。 为 了 从 右边 的 表 中 选择 所 有 行 , 需要 使 
用 RIGHT OUTER JOIN， 如 下 例 所 示 : 


输入 了 



























































SELECT Customers.cust_id, Orders.order_num 
FROM Customers RIGHT OUTER JOIN Orders 
ON Orders.cust_id = Customers.cust_id; 





注意 : SOLite 外 联结 

SQLite 支持 LEFT OUTER JOIN， 但 不 支持 RIGHT OUTER JOIN, #3, 
如 果 你 确实 需要 在 SQLite 中 使 用 RIGHT OUTER JOIN， 有 一 种 更 简单 
的 办 法 ， 这 将 在 下 面 的 提示 中 介绍 。 








提示 : 外 联结 的 类 型 

要 记 住 ， 总 是 有 两 种 基本 的 外 联结 形式 : 左 外 联结 和 右 外 联结 。 它 们 
之 间 的 唯一 差别 是 所 关联 的 表 的 顺序 。 换 句 话说 , 调整 FROM 或 WHERE 
子 句 中 表 的 顺序 ， 左 外 联结 可 以 转换 为 右 外 联结 。 因 此 ， 这 两 种 外 联 
结 可 以 互 换 使 用 ， 哪 个 方便 就 用 哪个 。 
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还 存在 另 一 种 外 联结 ， 就 是 全 外 联结 (full outer join ), J 
的 所 有 行 并 关联 那些 可 以 关联 的 行 。 与 左 外 联结 或 右 外 联结 包含 一 个 表 
的 不 关联 的 行 不 同 ， 全 外 联结 包含 两 个 表 的 不 关联 的 行 。 o 吉 的 语 
法 如 下 : 

















输入 了 


SELECT Customers.cust_id, Orders.order_num 
FROM Orders FULL OUTER JOIN Customers 
ON Orders.cust_id = Customers.cust_id; 





注意 : FULL OUTER JOIN 的 支持 


Access, MariaDB, MySQL. Open Office Base 和 SQLite 不 支持 FULL 
OUTER JOIN 语法 。 











13.3 ”使 用 带 聚 集 函 数 的 联结 


如 第 9 课 所 述 ， 聚 集 函 数 用 来 汇总 数据 。 虽 然 至 今 为 止 我 们 举 的 聚集 
函数 的 例子 都 只 是 从 一 个 表 中 汇总 数据 ， 但 这 些 函 数 也 可 以 与 联结 
起 使 用 。 











Tt 























我 们 来 看 个 例子 ， 要 检索 所 有 顾客 及 每 个 顾客 所 下 的 订单 数 ， 下 面 的 代 
码 使 用 COUNT O 函数 完成 此 工作 ; 





输入 了 


SELECT Customers .cust_id， 
COUNT(Orders.order_num) AS num_ord 

FROM Customers INNER JOIN Orders 

ON Customers.cust_id = Orders.cust_id 

GROUP BY Customers.cust_id; 
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吕 


输出 


cust_id num_ord 
1000000001 2 
1000000003 1 
1000000004 1 
1000000005 1 


BY 





这 条 SELECT 语句 使 用 INNER JOIN 将 Customers 和 Orders 表 互 相关 联 。 
GROUP BY 子 句 按 顾 客 分 组 数据 ,因此 , XOH COUNT (Orders .order_num) 
对 每 个 顾客 的 订单 计数 ， 将 它 作为 num_ord 返回 。 








聚集 函数 也 可 以 方便 地 与 其 他 联结 一 起 使 用 。 请 看 下 面 的 例子 : 


输入 了 


SELECT Customers .cust_id， 
COUNT(Orders.order_num) AS num_ord 

FROM Customers LEFT OUTER JOIN Orders 

ON Customers .cust_id = Orders.cust_id 

GROUP BY Customers .cust_id; 





提示 : Oracle 中 没有 AS 
再 次 提醒 Oracle 用 户 ， 请 记 住 删除 AS。 








输出 了 


Cust_id num_ord 
1000000001 2 
1000000002 0 
1000000003 1: 
1000000004 1 
1000000005 1 
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BY 


这 个 例子 使 用 左 外 部 联结 来 包含 所 有 顾客 ， 甚 至 包含 那些 没有 任何 订单 
的 顾客 。 结 果 中 也 包含 了 顾客 1000000002， 他 有 0 个 订单 。 


13.4 ”使 用 联结 和 联结 条 件 
在 总 结 讨论 联结 的 这 两 课 前 ， 有 必要 汇总 一 下 联结 及 其 使 用 的 要 点 。 


口 注意 所 使 用 的 联结 类 型 。 一 般 我 们 使 用 内 联结 , 但 使 用 外 联结 也 有 效 。 
O 关于 确切 的 联结 语法 , 应 该 查看 具体 的 文档 , 看 相应 的 DBMS 支持 何 
种 语法 ( 大 多 数 DBMS 使 用 这 两 课 中 描述 的 某 种 语法 )。 

O 保证 使 用 正确 的 联结 条 件 ( 不 管 采用 哪 种 语法 )， 否 则 会 返回 不 正确 
的 数据 。 

口 应 该 总 是 提供 联结 条 件 ， 否 则 会 得 出 第 卡 儿 积 。 

口 在 一 个 联结 中 可 以 包含 多 个 表 ， 其 至 可 以 对 每 个 联结 采用 不 同 的 联结 
类 型 。 虽 然 这 样 做 是 合法 的 ， 一 般 也 很 有 用 ， 但 应 该 在 一 起 测试 它们 
前 分 别 测 试 每 个 联结 。 这 会 使 故障 排除 更 为 简单 。 

















13.5 小结 

本 课 是 上 一 课 的 延续 ， 首 先 讲授 了 如 何以 及 为 什么 使 用 别名 ， 然 后 讨论 
不 同 的 联结 类 型 以 及 每 类 联结 所 使 用 的 语法 。 我 们 还 介绍 了 如 何 与 联结 
一 起 使 用 聚集 函数 ， 以 及 在 使 用 联结 时 应 该 注意 的 问题 。 
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本 课 讲述 如 何 利 用 UNION 操作 符 将 多 条 SELECT 语句 组 合成 一 个 结 


H. 
果 集 。 











14.1 组 合 查询 


多 数 SQL 查询 只 包含 从 一 个 或 多 个 表 中 返回 数据 的 单条 SELECT 语句 。 
但 是 ，SQL 也 允许 执行 多 个 查询 (多 条 SELECT 语句 )， 并 将 结果 作为 一 
个 查询 结果 集 返 回 。 这 些 组 合 查 询 通 常 称 为 并 ( union ) 或 复合 查询 
( compound query )。 























主要 有 两 种 情况 需要 使 用 组 合 查询 : 





Q 在 一 个 查询 中 从 不 同 的 表 返 回 结构 数据 ; 
口 对 一 个 表 执 行 多 个 查询 ， 按 一 个 查询 返回 数据 。 








提示 : 组 合 查询 和 多 个 WHERE 条 件 

多 数 情况 下 ， 组 合 相 同 表 的 两 个 查询 所 完成 的 工作 与 具有 多 个 WHERE 
子 句 条 件 的 一 个 查询 所 完成 的 工作 相同 。 换 向 话 说 ， 任 何 具 有 多 个 
WHERE Jú SELECT 语句 都 可 以 作为 一 个 组 合 查 询 ， 在 下 面 可 以 看 


到 BIS 点 ° 
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14.2 ”创建 组 合 查询 


可 用 UNION 操作 符 来 组 合 数 条 SQL 查询 。 利 用 UNION， 可 给 出 多 条 
SELECT 语句 ， 将 它们 的 结果 组 合成 一 个 结果 














Š 


° 


14.2.1 使 用 UNION 








使 用 UNION 很 简单 ， 所 要 做 的 只 是 给 出 每 条 SELECT 语句 ， 在 各 条 语句 
之 间 放 上 关键 字 UNION。 





举 个 例子 , 假如 需要 Illinois. Indiana 和 Michigan 等 美国 几 个 州 的 所 有 顾 
客 的 报表 ， 还 想 包括 不 管 位 于 哪个 州 的 所 有 的 Fun4A11。 当 然 可 以 利用 
WHERE 子 句 来 完成 此 工作 ， 不 过 这 次 我 们 使 用 UNION, 


如 上 所 述 , 创建 UNION 涉及 编写 多 条 SELECT 语句 。 首先 来 看 单条 语句 : 
输入 了 


SELECT cust_name, cust_contact, cust_email 
FROM Customers 
WHERE cust_state IN ('IL','IN','MI'); 


输出 了 

cust_name cust_contact cust_email 

Village Toys John Smith sales@villagetoys.com 
Fun4A11 Jim Jones jjonesQfun4a11.com 
The Toy Store Kim Howard NULL 

输入 了 


SELECT cust_name, cust_contact, cust_email 
FROM Customers 
WHERE cust_name = 'Fun4A11'; 
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输出 了 

cust_name cust_contact cust_email 
E jjönesafunsdi icon 
Fun4A11 Denise L. Stephens dstephens@fun4all.com 
SH Y 


第 一 条 SELECT 把 Illinois Indiana , Michigan 等 州 的 缩写 传递 给 IN FA, 
仿 索 出 这 些 州 的 所 有 行 。 第 二 条 SELECT 利用 简单 的 相等 测试 找 出 所 有 
Fun4A11。 





组 合 这 两 条 语句 ， 可 以 如 下 进行 : 


输入 了 


SELECT cust_name, cust_contact, cust_email 
FROM Customers 

WHERE cust_state IN ('IL','IN','MI') 

UNION 

SELECT cust_name, cust_contact, cust_email 
FROM Customers 

WHERE cust_name = 'Fun4All'; 


输出 了 

cust_name cust_contact cust_email 

Fun4A11 Denise L. Stephens dstephens@fun4all.com 
Fun4A11 Jim Jones jjones@fun4all.com 
Village Toys John Smith sales@villagetoys.com 
The Toy Store Kim Howard NULL 

SH Y 


这 条 语句 由 前 面 的 两 条 SELECT 语句 组 成 ， 之 间 用 UNION 关键 字 分 隔 。 
UNION 指示 DBMS 执行 这 两 条 SELECT 语句 ， 并 把 输出 组 合成 一 个 查询 
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结果 集 。 











为 了 便于 参考 ,这 里 给 出 使 用 多 条 WHERE 子 句 而 不 是 UNION 的 相同 查询 : 
输入 了 


SELECT cust_name, cust_contact, cust_email 
FROM Customers 

WHERE cust_state IN ('IL','IN','MI') 

OR cust_name = 'Fun4A11'; 








在 这 个 简单 的 例子 中 , 使 用 UNION 可 能 比 使 用 WHERE 子 句 更 为 复杂 。 但 
对 于 较 复 杂 的 过 滤 条 件 ， 或 者 从 多 个 表 ( 而 不 是 一 个 表 ) 中 检索 数据 的 
情形 ， 使 用 UNION 可 能 会 使 处 理 更 简单 。 
































提示 : UNION 的 限制 

使 用 UNION 组 合 SELECT 语句 的 数目 ，SQL 没有 标准 限制 。 但 是 ， 最 
好 是 参考 一 下 具体 的 DBMS 文档 , 了 解 它 是 否 对 UNION 能 组 合 的 最 大 
语句 数目 有 限制 。 








注意 : 性 能 问题 

DBMS 使 用 内 部 查询 优化 程序 ， 在 处 理 各 条 SELECT 语句 前 
合 它们 。 理论 上 讲 ， 这 意味 着 从 性 能 上 看 使 用 多 条 WHERE 子 句 条 件 

还 是 UNION 应 该 没有 实际 的 差别 。 不 过 我 说 的 是 理论 上 ， 实 践 中 多 数 

1 EE 达到 理想 状态 ， 所 以 最 好 测试 一 下 这 两 种 方法 ， 

看 哪 种 工作 得 更 











14.2.2 UNION 规则 





可 以 看 到 ，UNION 非常 容易 使 用 ， 但 在 进行 组 合 时 需要 注意 几 条 规则 。 
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T UNION 必须 由 两 条 或 两 条 以 上 的 SELECT 语句 组 成 ， 语 句 之 间 用 关键 
字 UNION 分 隔 ( 因此 ,如 果 组 合 四 条 SELECT 语句 ,将 要 使 用 三 个 UNION 
关键 字 )。 

口 UNION 中 的 每 个 查询 必须 包含 相同 的 列 、 表 达 式 或 聚集 函数 (不 过 ， 

各 个 列 不 需要 以 相同 的 次 序列 出 )。 

口 列 数据 类 型 必须 兼容 : 类 型 不 必 完 全 相同 , 但 必须 是 DBMS 可 以 隐 含 
转换 的 类 型 ( 例如 ， 不 同 的 数值 类 型 或 不 同 的 日 期 类 型 )。 

































































如 果 遵 守 了 这 些 基 本 规则 或 限制 ， 则 可 以 将 UNION 用 于 任何 数据 检索 
操作 。 


14.2.3 ”包含 或 取消 重复 的 行 


回 到 14.2.1 节 ， 我 们 看 看 所 用 的 SELECT 语句 。 注 意 到 在 分 别 执行 语句 
时 ,第 一 条 SELECT 语句 返回 3 行 , 第 二 条 SELECT 语句 返回 2 行 。 而 在 
用 UNION 组 合 两 条 SELECT 语句 后 ， 只 返回 4 行 而 不 是 5 行 。 














UNION 从 查询 结果 集中 自动 去 除了 重复 的 行 ; 换 句 话说 ， 它 的 行为 与 一 
条 SELECT 语句 中 使 用 多 个 WHERE 子 句 条 件 一 样 。 因 为 Indiana 州 有 一 个 
Fun4Al 单位 ,所 以 两 条 SELECT 语句 都 返回 该 行 。 使 用 UNION 时 , 重复 
的 行 会 被 自动 取消 。 

















这 是 UNION 的 默认 行为 ， 如 果 愿 意 也 可 以 改变 它 。 事 实 上 ， 如 果 想 返回 
所 有 的 匹配 行 ， 可 使 用 UNION ALL 而 不 是 UNION。 


请 看 下 面 的 例子 : 
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输入 了 


SELECT cust_name, cust_contact, cust_email 
FROM Customers 

WHERE cust_state IN ('IL','IN','MI') 

UNION ALL 

SELECT cust_name, cust_contact, cust_email 
FROM Customers 

WHERE cust_name = 'Fun4All'; 


输出 了 

cust_name cust_contact cust_email 

Village Toys John Smith sales@villagetoys.com 
Fun4A11 Jim Jones jjones@fun4all.com 
The Toy Store Kim Howard NULL 

Fun4A11 Jim Jones jjones@fun4all.com 
Fun4A11 Denise L. Stephens dstephens@fun4all.com 
SH Y 





使 用 UNION ALL, DBMS 不 取消 重复 的 行 。 因 此 ， 这 里 返回 5 行 ， 其 中 
有 一 行 出 现 两 次 。 





提示 : UNION 与 WHERE 

这 一 课 一 开始 我 们 说 过 ，UNION 几乎 总 是 完成 与 多 个 WHERE 条 件 相 同 
的 工作 。UNION ALL 为 UNION 的 一 种 形式 ， 它 完成 WHERE + ëJ Z X, 
不 了 的 工作 。 如 果 确 实 需要 每 个 条 件 的 匹配 行 全 部 出 现 ( 包括 重复 行 )， 
就 必须 使 用 UNION ALL， 而 不 是 WHERE。 











14.24 ”对 组 合 查询 结果 排序 

SELECT 语句 的 输出 用 ORDER BY 子 名 排序。 在 用 UNION 组 合 查询 时 ， 只 
能 使 用 一 条 ORDER BY 子 句 ， 它 必须 位 于 最 后 一 条 SELECT 语句 之 后 。 对 
于 结果 集 ， 不 存在 用 一 种 方式 排序 一 部 分 ， 而 又 用 另 一 种 方式 排序 另 一 
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部 分 的 情况 ， 因 此 不 允许 使 用 多 条 ORDER BY 子 句 。 

















下 面 的 例子 对 前 面 UNION 返回 的 结果 进行 排序 : 


输入 了 


SELECT cust_name, cust_contact, cust_email 
FROM Customers 

WHERE cust_state IN ('IL','IN','MI') 

UNION 

SELECT cust_name, cust_contact, cust_email 
FROM Customers 

WHERE cust_name = 'Fun4A11' 

ORDER BY cust_name, cust_contact; 


输出 了 

cust_name cust_contact cust_email 

Fun4A11 Denise L. Stephens dstephens@fun4all.com 
Fun4A11 Jim Jones jjones@fun4all.com 
The Toy Store Kim Howard NULL 

Village Toys John Smith sales@villagetoys.com 
SH Y 





这 条 UNION 在 最 后 一 条 SELECT 语句 后 使 用 了 ORDER BY 子 句 。 虽 然 ORDER 
BY 子 句 似乎 只 是 最 后 一 条 SELECT 语句 的 组 成 部 分 , 但 实际 上 DBMS 将 
用 它 来 排序 所 有 SELECT 语句 返回 的 所 有 结 











说 明 : 其 他 类 型 的 UNION 

某 些 DBMS 还 支持 另外 两 种 UNION: EXCEPT ( 有 时 称 为 MINUS ) 可 用 
来 检索 只 在 第 一 个 表 中 存在 而 在 第 二 个 表 中 不 存在 的 行 ; 而 
INTERSECT 可 用 来 检索 两 个 表 中 都 存在 的 行 。 实际 上 , 这 些 UNION 很 
少 使 用 ， 因 为 相同 的 结果 可 利用 联结 得 到 。 
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提示 : 操作 多 个 表 

为 了 简单 ， 本 课 中 的 例子 都 是 使 用 UNION 来 组 合 针对 同一 表 的 多 个 查 
询 。 实 际 上 ，UNION 在 需要 组 合 多 个 表 的 数据 时 也 很 有 用 ， 即 使 是 有 
不 匹配 列 名 的 表 ， 在 这 种 情况 下 ， 可 以 将 UNION 与 别名 组 合 ， 检 索 
一 个 结果 集 。 








14.3 小结 


这 一 课 讲 授 如 何 用 UNION 操作 符 来 组 合 SELECT 语句 。 利 用 UNION, nj 
以 把 多 条 查询 的 结果 作为 一 条 组 合 查 询 返 回 ， 不 管 结果 中 有 无 重复 。 使 
用 UNION 可 极 大 地 简化 复杂 的 WHERE 子 句 , 简化 从 多 个 表 中 检索 数据 的 
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这 一 课 介绍 如 何 利用 SQL 的 INSERT 语句 将 数据 插入 表 中 。 


15.1 数据 插入 

ZJ EN, SELECT 是 最 常用 的 SQL 语句 了 ， 这 就 是 前 14 课 都 在 讲 它 的 
原因 。 但 是 ,还 有 其 他 3 个 常用 的 SQL 语句 需要 学 习 。 第 一 个 就 是 INSERT 
(下 一 课 介绍 另外 两 个 )。 


























顾名思义 ，INSERT 用 来 将 行 插 入 ( 或 添加 ) 到 数据 库 表 。 插 和信 有 几 种 
方式 : 


























提示 : 插入 及 系统 安全 
使 用 INSERT 语句 可 能 需要 客户 端 /服务 器 DBMS 中 的 特定 安全 权限 。 
在 你 试图 使 用 INSERT 前 ， 应 该 保证 自己 有 足够 的 安全 权限 。 
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15.1.1 插入 完整 的 行 


把 数据 插入 表 中 的 最 简单 方法 是 使 用 基本 的 INSERT 语法 ， 它 要 求 指定 
表 名 和 插入 到 新 行 中 的 值 。 下 面 举 一 个 例子 : 



































输入 了 


INSERT INTO Customers 
VALUES('1000000006' , 
'Toy Land', 
"123 Any Street ', 
"New York', 
'NY', 
"113411" 
'USA', 
NULL, 
NULL); 


BY 


这 个 例子 将 一 个 新 顾客 插入 到 Customers 表 中 。 存储 到 表 中 每 一 列 的 数 
据 在 VALUES 子 句 中 给 出 ， 必 须 给 每 一 列 提供 一 个 值 。 如 果 某 列 没 有 值 ， 
如 上 面 的 cust_contact 和 cust_emai1 列 , 则 应 该 使 用 NULL 值 (假定 
表 人 允许 对 该 列 指定 空 值 )。 各 列 必须 以 它们 在 表 定 义 中 出 现 的 次 序 填充 。 














T 








提示 : INTO 关键 字 

在 某 些 SQL 实现 中 ， 跟 在 INSERT 之 后 的 INTO 关键 字 是 可 选 的 。 但 
是 ， 即 使 不 一 定 需要 ， 最 好 还 是 提供 这 个 关键 字 ， 这 样 做 将 保证 SQL 
代码 在 DBMS 之 间 可 移植 。 











虽然 这 种 语法 很 简单 ， 但 并 不 安全 ,应 该 尽量 避免 使 用 。 上 面 的 SQL i 
句 高 度 依赖 于 表 中 列 的 定义 次 序 ， 还 依赖 于 其 容易 获得 的 次 序 信息 。 即 
使 可 以 得 到 这 种 次 序 信息 ， 也 不 能 保证 各 列 在 下 一 次 表 结 构 变 动 后 保持 
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完全 相同 的 次 序 。 因 此 ， 编 写 依赖 于 特定 列 次 序 的 SQL 语句 是 很 不 安全 
的 ， 这 样 做 迟早 会 出 问题 。 








编写 INSERT 语句 的 更 安全 (不 过 更 烦琐 ) 的 方法 如 下 : 


输入 了 


INSERT INTO Customers(cust_id, 
cust_name, 
cust_address, 
cust_city, 
cust_state, 
Cust_zip， 
cust_country, 
cust_contact, 
cust_email) 

VALUES('1000000006' , 

'Toy Land', 
"123 Any Street ' ， 
"New York ' ， 


BY 


这 个 例子 与 前 一 个 INSERT 语句 的 工作 完全 相同 ， 但 在 表 名 后 的 括号 里 
明确 给 出 了 列 名 。 在 插入 行 时 ，DBMS 将 用 VALUES 列表 中 的 相应 值 填 
入 列表 中 的 对 应 项 。VALUES 中 的 第 一 个 值 对 应 于 第 一 个 指定 列 名 , 第 二 
个 值 对 应 于 第 二 个 列 名 ， 如 此 等 等 。 























因为 提供 了 列 名 ,VALUES 必须 以 其 指定 的 次 序 匹配 指定 的 列 名 , 不 一 定 
按 各 列 出 现在 表 中 的 实际 次 序 。 其 优点 是 ， 即 使 表 的 结构 改变 ， 这 条 
INSERT 语句 仍然 能 正确 工作 。 
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下 面 的 INSERT 语句 填充 所 有 列 〈 与 前 面 的 一 样 )， 但 以 一 种 不 同 的 次 序 
填充 。 因 为 给 出 了 列 名 ， 所 以 插入 结果 仍然 正确 : 




















输入 了 


INSERT INTO Customers(cust_id, 
cust_contact, 
cust_email, 


cust_name, 
cust_address, 
cust_city, 
cust_state, 
cust_zip) 
VALUES('1000000006' , 

NULL, 

NULL, 

'Toy Land', 

"123 Any Street ' ， 

"New York ' ， 

'NY', 

'11111'); 





提示 : 总 是 使 用 列 的 列表 
不 要 使 用 没有 明确 给 出 列 的 INSERT 语句 。 给 出 列 能 使 SQL 代码 继续 
发 挥 作用 ， 即 使 表 结 构 发 生 了 变化 。 








注意 : 小 心 使 用 VALUES 

不 管 使 用 哪 种 INSERT 语法 , VALUES 的 数目 都 必须 正确 。 如 果 不 提供 列 
名 ， 则 必须 给 每 个 表 列 提供 一 个 值 ， 如果 提供 列 名 , 则 必须 给 列 出 的 每 
个 列 一 个 值 。 否 则 ， 就 会 产生 一 条 错误 消息 ， 相 应 的 行 不 能 成 功 插入 。 











15.1.2 插入 部 分 行 

















正如 所 述 ， 使 用 INSERT 的 推荐 方法 是 明确 给 出 表 的 列 名 。 使 用 这 种 
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语法 ， 还 可 以 省 略 列 ， 这 表示 可 以 只 给 某 些 列 提 供 值 ， 给 其 他 列 不 提 
供 值 。 





请 看 下 面 的 例子 : 

输入 

INSERT INTO Customers(cust_id, 
cust_name, 
cust_address, 
cust_city, 
cust_state, 
Cust_zip， 


cust_country) 

VALUES('1000000006' , 

'Toy Land', 

"123 Any Street ' ， 

"New York ' ， 

'NY', 

'11111', 

'USA'); 


BY 


在 本 课 前 面 的 例子 中 ， 没 有 给 cust contact 和 cust email 这 两 列 提 
供 值 。 这 表示 没 必 要 在 INSERT 语句 中 包含 它们 。 因 此 ， 这 里 的 INSERT 
语句 省 略 了 这 两 列 及 其 对 应 的 值 。 











注意 : 省 略 列 
如 果 表 的 定义 允许 ， 则 可 以 在 INSERT 操作 中 省 略 某 些 列 。 省 略 的 列 
必须 满足 以 下 某 个 条 件 。 


O 该 列 定义 为 允许 NULL 值 (无 值 或 空 值 )。 
口 在 表 定 义 中 给 出 默认 值 。 这 表示 如 果 不 给 出 值 ， 将 使 用 默认 值 。 
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注意 : 省 略 所 需 的 值 
如 果 表 中 不 允许 有 NULL 值 或 者 默认 值 ， 这 时 却 省 略 了 表 中 的 值 ， 
DBMS 就 会 产生 错误 消息 ， 相 应 的 行 不 能 成 功 插入 。 











15.1.3 ”插入 检索 出 的 数据 


INSERT 一 般 用 来 给 表 插入 具有 指定 列 值 的 行 。INSERT 还 存在 另 一 种 
形式 ， 可 以 利用 它 将 SELECT 语句 的 结果 插入 表 中 ， 这 就 是 所 谓 的 
INSERT SELECT。 顾 名 思 义 ， 它 是 由 一 条 INSERT 语句 和 一 条 SELECT 
语句 组 成 的 。 












































腿 如 想 把 另 一 表 中 的 顾客 列 合 并 到 Customers 表 中 。 不 需要 每 次 读 取 一 


Z 


了 再 将 它 用 INSERT 插入 ， 可 以 如 下 进行 : 


























输入 了 


INSERT INTO Customers(cust_id, 
cust_contact, 
cust_email, 
cust_name, 
cust_address, 
cust_city, 
cust_state, 
cust_zip, 
cust_country) 

SELECT cust_id, 

cust_contact, 
cust_email, 
cust_name, 
cust_address, 
cust_city, 
cust_state, 
cust_zip, 
cust_country 

FROM CustNew; 
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说 明 : 新 例子 的 说 明 

这 个 例子 从 一 个 名 为 CustNew 的 表 中 读 出 数据 并 插入 到 Customers 
表 。 为 了 试验 这 个 例子 ， 应 该 首先 创建 和 填充 CustNew 表 。CustNew 
表 的 结构 与 附录 A 中 描述 的 Customers 表 相 同 。 在 填充 CustNew 时 ， 
不 应 该 使 用 已 经 在 Customers 中 用 过 的 cust_id 值 (如 果 主 键 值 重 
复 ， 后续 的 INSERT 操作 将 会 失败 )。 








BY 


这 个 例子 使 用 INSERT SELECT 从 CustNew 中 将 所 有 数据 导入 
Customers。SELECT 语句 从 CustNew 检索 出 要 插 人 的 值 ， 而 不 是 列 出 
它们 。SELECT 中 列 出 的 每 一 列 对 应 于 Customers 表 名 后 所 跟 的 每 一 列 。 
这 条 语句 将 插入 多 少 行 呢 ? 这 依赖 于 CustNew 表 有 多 少 行 。 如 果 这 个 表 
为 空 ， 则 没有 行 被 插入 (也 不 产生 错误 ， 因 为 操作 仍然 是 合法 的 )。 如 果 
这 个 表 确 实 有 数据 ， 则 所 有 数据 将 被 插入 到 Customers。 























提示 : INSERT SELECT 中 的 列 名 

为 简单 起 见 ,这 个 例子 在 INSERT 和 SELECT 语 多 中 使 用 了 相同 的 列 名 。 
但 是 , 不 一 定 要 求 列 名 匹配 。 事实 上 , DBMS 一 点 儿 也 不 关心 SELECT 
返回 的 列 名 。 它 使 用 的 是 列 的 位 置 ， 因 此 SELECT 中 的 第 一 列 (不 管 
其 列 名 ) 将 用 来 填充 表 列 中 指定 的 第 一 列 ， 第 二 列 将 用 来 填充 表 列 中 


A 


指定 的 第 二 列 ， 如 此 等 等 。 








INSERT SELECT 中 SELECT 语句 可 以 包含 WHERE 子 句 ， 以 过 滤 插 入 的 数据 。 








提示 : 插入 多 行 

INSERT 通常 只 插入 一 行 。 要 插入 多 行 ， 必 须 执行 多 个 INSERT 语句 。 
INSERT SELECT 有 是 个 例外 , 它 可 以 用 一 条 INSERT 播 入 多 行 , 不 管 SELECT 
语句 返回 多 少 行 ， 都 将 被 INSERT 插入 。 
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15.2 ”从 一 个 表 复 制 到 另 一 个 表 


有 一 种 数据 插入 不 使 用 INSERT 语句 。 要 将 一 个 表 的 内 容 复 制 到 一 个 全 
新 的 表 ( 运行 中 创建 的 表 )， 可 以 使 用 SELECT INTO 语句 。 











说 明 : DB2 不 支持 
DB2 不 支持 这 里 描述 的 SELECT INTO。 











与 INSERT SELECT 将 数据 添加 到 一 个 已 经 存在 的 表 不 同 ，SELECT INTO 
将 数据 复制 到 一 个 新 表 (有 的 DBMS 可 以 覆盖 已 经 存在 的 表 ， 这 依赖 于 
所 使 用 的 具体 DBMS )。 











说 明 : INSERT SELECT 与 SELECT INTO 
它们 之 间 的 一 个 重要 差别 是 前 者 导出 数据 ， 而 后 者 导入 数据 。 











下 面 的 例子 说 明 如 何 使 用 SELECT INTO: 


输入 了 


SELECT * 
INTO CustCopy 
FROM Customers ; 


BY 


这 条 SELECT 语句 创建 一 个 名 为 CustCopy 的 新 表 ， 并 把 Customers 表 
的 整个 内 容 复制 到 新 表 中 。 因 为 这 里 使 用 的 是 SELECT * ， 所 以 将 在 
CustCopy 表 中 创建 ( 并 填充 ) 与 Customers 表 的 每 一 列 相同 的 列 。 要 
想 只 复制 部 分 的 列 ， 可 以 明确 给 出 列 名 ， 而 不 是 使 用 * 通 配 符 。 
























































MariaDB, MySQL, Oracle, PostgreSQL 和 SQLite 使 用 的 语法 稍 有 不 同 : 
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输入 了 


CREATE TABLE CustCopy AS 
SELECT * FROM Customers ; 


在 使 用 SELECT INTO 时 ， 需 要 知道 一 些 事情 : 





口 任何 SELECT 选项 和 子 句 都 可 以 使 用 ， 包 括 WHERE 和 GROUP BY; 
a 可 利用 联结 从 多 个 表 插 入 数据 ; 
a 不 管 从 多 少 个 表 中 检索 数据 ， 数 据 都 只 能 搬入 到 一 个 表 中 。 

















提示 : 进行 表 的 复制 
SELECT INTO 是 试验 新 SQL 语句 前 进行 表 复 制 的 很 好 工具 。 先 进行 复 
制 ， 可 在 复制 的 数据 上 测试 SQL 代码 ， 而 不 会 影响 实际 的 数据 。 








说 明 : 更 多 例子 
如 果 想 看 INSERT 用 法 的 更 多 例子 ， 请 参阅 附录 A 中 给 出 的 样 例 表 填 
充 脚 本 。 











15.3 小结 


这 一 课 介绍 如 何 将 行 插入 到 数据 库 表 中 。 我 们 学 习 了 使 用 INSERT 的 几 
种 方法 ,为 什么 要 明确 使 用 列 名 ， 如何 用 INSERT SELECT 从 其 他 表 中 导 
入行， 如 何 用 SELECT INTO 将 行 导出 到 一 个 新 表 。 下 一 课 将 讲述 如 何 使 
用 UPDATE 和 DELETE 进一步 操作 表 数 据 。 
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这 一 课 介绍 如 何 利 用 UPDATE 和 DELETE 语句 进一步 操作 表 数 据 。 





16.1 更 新 数据 

更 新 (修改 ) 表 中 的 数据 ， 可 以 使 用 UPDATE 语句。 有 两 种 使 用 UPDATE 
的 方式 : 

D 更 新 表 中 的 特定 行 ; 

口 更 新 表 中 的 所 有 行 。 











下 面 分 别 介 绍 。 





注意 : 不 要 省 略 WHERE +J 
在 使 用 UPDATE 时 一 定 要 细心 。 因 为 稍 不 注意 ， 就 会 更 新 表 中 的 所 有 
行 。 使 用 这 条 语句 前 ， 请 完整 地 阅读 本 节 。 








提示 : UPDATE 与 安全 
在 客户 端 /服务 器 的 DBMS 中 , 使 用 UPDATE 语句 可 能 需要 特殊 的 安全 
权限 。 在 你 使 用 UPDATE 前 ， 应 该 保证 自己 有 足够 的 安全 权限 。 

















使 用 UPDATE 语句 非常 容易 , 甚至 可 以 说 太 容易 了 。 基本 的 UPDATE 语句 
由 三 部 分 组 成 ， 分 别 是 : 
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on 





口 要 更 新 的 表 ; 
口 列 名 和 它们 的 新 值 ; 
口 确定 要 更 新 哪些 行 的 过 滤 条 件 。 

















举 一 个 简单 例子 。 客 户 1000000005 现在 有 了 电子 邮件 地 址 ， 因 此 他 的 
记录 需要 更 新 ,语句 如 下 : 








输入 了 


UPDATE Customers 
SET cust_email = 'kim@thetoystore.com' 
WHERE cust_id = '1000000005'; 





UPDATE 语句 总 是 以 要 更 新 的 表 名 开始 。 在 这 个 例子 中 , 要 更 新 的 表 名 为 
Customers。SET 命令 用 来 将 新 值 赋 给 被 更 新 的 列 。 在 这 里 ，SET 子 句 设 
置 cust_email 列 为 指定 的 值 : 











SET cust_email = 'kim@thetoystore.com' 
UPDATE 语句 以 WHERE 子 句 结束 , 它 告诉 DBMS 更 新 哪 一 行 。 没 有 WHERE 
FAJ, DBMS 将 会 用 这 个 电子 邮件 地 址 更 新 Customers 表 中 的 所 有 行 ， 
这 不 是 我 们 希望 的 。 





























更 新 多 个 列 的 语法 稍 有 不 同 : 





输入 了 

UPDATE Customers 

SET cust_contact = 'Sam Roberts ' ， 
cust_email = 'sam@toyland.com' 


WHERE cust_id = '1000000006'; 


在 更 新 多 个 列 时 ， 只 需要 使 用 一 条 SET 命令 ， 每 个 “ 列 = 值 ” 对 之 间 用 
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逗号 分 隔 ( 最 后 一 列 之 后 不 用 逗号 ), 在 此 例子 中 , 更 新 顾客 1000000006 


的 cust_contact 和 cust_emai1 列 。 











提示 : 在 UPDATE 语句 中 使 用 子 查询 
UPDATE 语句 中 可 以 使 用 子 查询 , 使 得 能 用 SELECT 语句 检索 出 的 数据 
更 新 列 数据 。 关 于 子 查询 及 使 用 的 更 多 内 容 ， 请 参阅 第 11 课 。 








提示 : FROM 关键 字 

有 的 SQL 实现 支持 在 UPDATE 语句 中 使 用 FROM 子 句 ， 用 一 个 表 的 数 
据 更 新 另 一 个 表 的 行 。 如 想 知 道 你 的 DBMS 是 否 支 持 这 个 特性 ， 请 参 
阅 它 的 文档 。 











要 删除 某 个 列 的 值 ， 可 设置 它 为 NULL ( 假如 表 定 义 允 许 NULL 值 )。 如 下 
进行 : 

输入 了 

UPDATE Customers 


SET cust_email = NULL 
WHERE cust_id = '1000000005'; 


HP NULL 用 来 去 除 cust_emai1 列 中 的 值 . 这 与 保存 空 字符 串 很 不 同 ( = 
字符 串 用 ' 表 示 ， 是 一 个 值 )， 而 NULL 表示 没有 值 。 





16.2 ”删除 数据 


从 一 个 表 中 删除 ( 去 掉 ) 数据 ,使 用 DELETE 语句 。 有 两 种 使 用 DELETE 
的 方式 : 





口 从 表 中 删除 特定 的 行 ; 
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口 从 表 中 删除 所 有 行 。 





注意 : 不 要 省 略 WHERE FA 
在 使 用 DELETE 时 一 定 要 细心 。 因 为 稍 不 注意 ， 就 会 错误 地 删除 表 中 
所 有 行 。 在 使 用 这 这 条 语句 前 A) 9 请 请 完整 地 阅读 本 节 ° 








提示 : DELETE 与 安全 
在 客户 端 /服务 器 的 DBMS 中 , 使 用 DELETE 语句 可 能 需要 特殊 的 安全 
权限 。 在 你 使 用 DELETE 前 ， 应 该 保证 自己 有 足够 的 安全 权限 。 














前 面 说 过 ，UPDATE 非常 容易 使 用 ， 而 DELETE 更 容易 使 用 





G 





下 面 的 语句 从 Customers 表 中 删除 一 行 


输入 了 


DELETE FROM Customers 
WHERE cust_id = '1000000006'; 





这 条 语句 很 容易 理解 。DELETE FROM 要 求 指定 从 中 删除 数据 的 表 名 ， 
WHERE 子 句 过 滤 要 删除 的 行 。 在 这 个 例子 中 ， 只 删除 顾客 1000000006. 
如 果 省 略 WHERE 子 句 ， 它 将 删除 表 中 每 个 顾客 。 






































提示 : 友好 的 外 键 

第 12 课 介绍 了 联结 ， 简 单 联结 两 个 表 只 需要 这 两 个 表 中 的 常用 字段 。 
也 可 以 让 DBMS 通过 使 用 外 键 来 严格 实施 关系 (这 些 定义 在 附录 A 
中 )。 存 在 外 键 时 ，DBMS 使 用 它们 实施 引用 完整 性 。 例 如 要 向 
Products 表 中 插入 一 个 新 产品 ，DBMS 不 允许 通过 未 知 的 供应 商 id 
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插入 它 ， 因 为 vendid 列 是 作为 外 键 连接 到 Vendors 表 的 。 那 么 ， 
这 与 DELETE 有 什么 关系 呢 ? 使 用 外 键 确 保 引 用 完整 性 的 一 个 好 处 是 ， 
DBMS 通常 可 以 防止 删除 某 个 关系 需要 用 到 的 行 。 例 如 ， 要 从 
Products 表 中 删除 一 个 产品 ,而 这 个 产品 用 在 OrderItems 的 已 有 订 
ŽP, ARA DELETE 语 多 将 抛 出 错误 并 中 止 。 这 是 总 要 定义 外 键 的 另 
一 个 理由 。 








提示 : FROM 关键 字 

在 某 些 SQL 实现 中 ， 跟 在 DELETE 后 的 关键 字 FROM 是 可 选 的 。 但 是 
即使 不 需要 ,也 最 好 提供 这 个 关键 字 。 这 样 做 将 保证 SQL 代 码 在 DBMS 
之 间 可 移植 。 











DELETE 不 需要 列 名 或 通配符 。DELETE 删除 整 行 而 不 是 删除 列 。 要 删除 
指定 的 列 ， 请 使 用 UPDATE 语句 。 








说 明 : 删除 表 的 内 容 而 不 是 表 
DELETE 语句 从 表 中 删除 行 ， 甚 至 是 删除 表 中 所 有 行 。 但 是 ，DELETE 
不 删除 表 本 身 。 








提示 : 更 快 的 删除 
如 果 想 从 表 中 删除 所 有 行 ,不 要 使 用 DELETE。 可 使 用 TRUNCATE TABLE 
语句 ， 它 完成 相同 的 工作 ， 而 速度 更 快 〈 因 为 不 记录 数据 的 变动 )。 











16.3 更 新 和 删除 的 指导 原则 


前 两 节 使 用 的 UPDATE 和 DELETE 语句 都 有 WHERE 子 句 ， 这 样 做 的 理由 
很 充分 。 如 果 省 略 了 WHERE fJ, W) UPDATE 或 DELETE 将 被 应 用 到 表 
中 所 有 的 行 。 换 句 话 说， 如 果 执 行 UPDATE 而 不 带 WHERE FAJ, MÆ} 
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每 一 行 都 将 用 新 值 更 新 。 类 似 地 ， 如 果 执行 DELETE 语句 而 不 带 WHERE 
子 句 ， 表 的 所 有 数据 都 将 被 删除 。 





























下 面 是 许多 SQL 程序 员 使 用 UPDATE 或 DELETE 时 所 遵循 的 重要 原则 。 





口 除非 确实 打算 更 新 和 删除 每 一 行 , 否则 绝对 不 要 使 用 不 带 WHERE FAJ 
的 UPDATE 或 DELETE 语句 。 

D 保证 每 个 表 都 有 主键 (如果 忘记 这 个 内 容 ， 请 参阅 第 12 课 )， 尽 可 能 
像 WHERE 子 句 那样 使 用 它 〈 可 以 指定 各 主键 、 多 个 值 或 值 的 范围 )。 
口 在 UPDATE 或 DELETE 语句 使 用 WHERE 子 句 前 ， 应 该 先 用 SELECT 进 
行 测试 , 保证 它 过 滤 的 是 正确 的 记录 , 以 防 编写 的 WHERE 子 句 不 正确 。 
口 使 用 强制 实施 引用 完整 性 的 数据 库 ( 关于 这 个 内 容 , 请 参阅 第 12 VË )， 
这 样 DBMS 将 不 允许 删除 其 数据 与 其 他 表 相 关联 的 行 。 

口 有 的 DBMS 允许 数据 库 管 理 员 施加 约束 ， 防 止 执行 不 带 WHERE FA 
的 UPDATE 或 DELETE 语句。 如 果 所 采用 的 DBMS 支持 这 个 特性 ， 应 
该 使 用 它 。 


















































若是 SQL 没有 撤销 (undo ) 按 钮 , 应 该 非常 小 心地 使 用 UPDATE 和 DELETE, 
否则 你 会 发 现 自己 更 新 或 删除 了 错误 的 数据 。 


16.4 小结 


这 一 课 讲述 了 如 何 使 用 UPDATE 和 DELETE 语句 处 理 表 中 的 数据 ,我 们 学 
习 了 这 些 语句 的 语法 ,知道 了 它们 可 能 存在 的 危险 , 了解 了 为 什么 WHERE 
子 句 对 UPDATE 和 DELETE 语句 很 重要 , 还 学 习 了 为 保证 数据 安全 而 应 该 
遵循 的 一 些 指导 原则 。 
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这 一 课 讲授 创建 、 更 改 和 删除 表 的 基 


第 17 课 创建 和 操纵 表 


17.1 创建 表 


SQL MXH 
本 身 的 创建 和 处 理 。 














一 般 有 两 种 创建 表 的 方法 : 





B 














本 知识 。 


日 于 表 数 据 操纵 ， 还 用 来 执行 数据 库 和 表 的 所 有 操作 ， 包 括 表 


O 多 数 DBMS 都 具有 交互 式 创建 和 管理 数据 库 表 的 工具 ; 
表 也 可 以 直接 用 SQL 语句 操纵 。 








用 程序 创建 表 ， 可 以 使 用 SQL 的 CREATE TABLE 语句 。 需 要 注意 的 是 , 使 


用 交互 式 工具 时 实际 
面 工具 会 














上 就 是 使 用 SQL 语句 。 这 些 语句 不 是 用 户 编写 的 , 界 
会 自动 生成 并 执行 相应 的 SQL 语句 (更改 已 有 的 表 时 也 是 这 样 )。 



































注意 : 语法 差别 
在 不 同 的 SQL 实现 中 ，CREATE TABLE 语句 的 语法 可 能 有 所 不 同 。 对 
于 具体 的 DBMS 支持 何 种 语法 ， 请 参阅 相应 的 文档 。 











这 一 课 不 会 介绍 创建 表 时 可 以 使 用 的 所 有 选项 ， 那 超出 了 本 课 的 范围 ， 
具体 的 DBMS 文档 。 


RI 





只 给 出 一 些 基 本 选项 。 





详细 的 信 
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说 明 : DBMS 创建 表 的 具体 例子 
关于 DBMS 的 CREATE TABLE 语句 的 具体 例子 , 请 参阅 附录 A 中 给 出 
的 样 例 表 创 建 脚本 。 











17.1.1 表 创 建 基础 
利用 CREATE TABLE 创建 表 ， 必 须 给 出 下 列 信 息 : 





口 新 表 的 名 字 ， 在 关键 字 CREATE TABLE 之 后 给 出 ; 
口 表 列 的 名 字 和 和 定义， 用 逗号 分 隔 ; 
口 有 的 DBMS 还 要 求 指 定 表 的 位 置 。 











下 面 的 SQL 语句 创建 本 书 中 所 用 的 Products K: 


输入 

CREATE TABLE Products 

( 
prod_id CHAR(10) NOT NULL, 
vend_id CHAR(10) NOT NULL, 
prod_name CHAR(254) NOT NULL, 
prod_price DECIMAL(8,2) NOT NULL, 
prod_desc VARCHAR(1000) NULL 

); 

S Y 





从 上 面 的 例子 可 以 看 到 ， 表 名 紧 跟 CREATE TABLE 关键 字 。 实 际 的 表 定 
X (所 有 列 ) 括 在 圆 括 号 之 中 ， 各 列 之 间 用 逗号 分 隔 。 这 个 表 由 5 列 组 
成 。 每 列 的 定义 以 列 名 ( 它 在 表 中 必须 是 唯一 的 ) 开始 ， 后 跟 列 的 数据 
类 型 ( 关于 数据 类 型 的 解释 ， 请 参阅 第 1 课 。 此 外 ， 附 录 D 列 出 了 和 常见 
的 数据 类 型 及 兼容 性 )。 整 条 语句 以 圆 括号 后 的 分 号 结束 。 
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前 面 提 到 ， 不 同 DBMS 的 CREATE TABLE 的 语法 有 所 不 同 ， 这 个 简单 脚 
本 也 说 明了 这 一 点 。 这 条 语句 在 Oracle PostgreSQL, .SQL Server 和 SQLite 
中 有 效 ， 而 对 于 MySQL, varchar 必须 替换 为 text; 对 于 DB2， 必 须 
从 最 后 一 列 中 去 掉 NULL, 这 就 是 对 于 不 同 的 DBMS, 要 编写 不 同 的 表 创 
建 脚本 的 原因 (参见 附录 A )。 























提示 : 语句 格式 化 

回想 一 下 在 SQL 语句 中 忽略 的 空格 。 语句 可 以 在 一 个 长 行 上 输入 ,也 
可 以 分 成 许多 行 ， 它 们 没有 差别 。 这 样 ， 你 就 可 以 用 最 适合 自己 的 方 
式 安 排 语句 的 格式 。 前 面 的 CREATE TABLE 语句 就 是 SQL 语句 格式 化 
的 一 个 好 例子 ， 代 码 安 排 在 多 个 行 上 ， 列 定义 进行 了 恰当 的 缩 进 ， 更 
易 阅 读 和 编辑 。 以 何 种 格式 安排 SQL 语句 并 没有 规定 ,但 我 强烈 推荐 
采用 某 种 缩 进 格式 。 








提示 : 共 换 现 有 的 表 
在 创建 新 的 表 时 ， 指 定 的 表 名 必须 不 存在 ， 否 则 会 出 错 。 防 止 意外 履 
盖 已 有 的 表 ，SQL 要 求 首先 手工 删除 该 表 (请 参阅 后 面 的 内 容 )， 然 
后 再 重建 它 ， 而 不 是 简单 地 用 创建 表 语句 复 盖 它 。 











17.1.2 ”使 用 NULL 值 

第 4 课 提 到 ，NULL 值 就 是 没有 值 或 缺 值 。 人 允许 NULL 值 的 列 也 允许 在 插 
入 行 时 不 给 出 该 列 的 值 。 不 允许 NULL 值 的 列 不 接受 没有 列 值 的 行 ， 换 
句 话 说， 在 插入 或 更 新 行 时 ， 该 列 必须 有 值 。 















































每 个 表 列 要 么 是 NULL ZJ, 要 么 是 NOT NULL 列 ， 这 种 状态 在 创建 时 由 表 
的 定义 规定 。 请 看 下 面 的 例子 : 
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输入 

CREATE TABLE Orders 

( 
order_num INTEGER NOT NULL, 
order_date DATETIME NOT NULL, 
cust_id CHAR(10) NOT NULL 

) ; 

SH Y 





这 条 语句 创建 本 书 中 所 用 的 Orders Ro Orders 包含 三 列 : 订单 号 、 订 
单 日 期 和 顾客 ID。 这 三 列 都 需要 ， 因 此 每 一 列 的 定义 都 含有 关键 字 NOT 
NULL, 这 就 会 阻止 插入 没有 值 的 列 。 如 果 插 入 没有 值 的 列 , 将 返回 错误 ， 
HARK. 

















下 一 个 例子 将 创建 混合 了 NULL 和 NOT NULL 列 的 表 : 


输入 了 

CREATE TABLE Vendors 

( 
vend_id CHAR(10) NOT NULL, 
vend_name CHAR(50) NOT NULL, 
vend_address CHAR(50) š 
vend_city CHAR(50) $ 
vend_state CHAR(5) j 
vend_zip CHAR(10) y 
vend_country CHAR(50) 

); 

5 Y 








这 条 语句 创建 本 书 中 使 用 的 Vendors 表 。 供 应 商 ID 和 供应 商 名 字 列 是 必 
需 的 , 因此 指定 为 NOT NULL。 其 余 五 列 全 都 允许 NULL 值 , 所 以 不 指定 NOT 
NULL, NULL 为 默认 设置 ， 如 果 不 指定 NOT NULL， 就 认为 指定 的 是 NULL 
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注意 : 指定 NULL 

在 不 指定 NOT NULL 时 ， 多 数 DBMS 认为 指定 的 是 NULL, 但 不 是 所 有 
的 DBMS 都 这 样 。 某 些 DBMS 要 求 指 定 关键 字 NULL， 如 果 不 指定 将 
出 错 。 关 于 完整 的 语法 信息 ， 请 参阅 具体 的 DBMS 文档 。 








提示 : 主键 和 NULL 值 
第 1 课 介绍 过 ,主键 是 其 值 唯一 标识 表 中 每 一 行 的 列 。 只 有 不 允许 NULL 
值 的 列 可 作为 主键 ， 允 许 NULL 值 的 列 不 能 作为 唯一 标识 。 








注意 : 理解 NULL 

不 要 把 NULL 值 与 空 字符 串 相 混淆 。NULL 值 是 没有 值 , 不 是 空 字符 串 。 
如 果 指 定 ''( 两 个 单 引 号 ,其 间 没 有 字符 )， 这 在 NOT NULL 列 中 是 允 
许 的 。 空 字符 囊 是 一 个 有 效 的 值 , 它 不 是 无 值 , NULL 值 用 关键 字 NULL 











17.1.3 ”指定 默认 值 


SQL 允许 指定 默认 值 ， 在 插入 行 时 如 果 不 给 出 值 ，DBMS 将 自动 采用 默 
认 值 ,默认 值 在 CREATE TABLE 语句 的 列 定 义 中 用 关键 字 DEFAULT 指定 。 
































请 看 下 面 的 例子 : 
输入 
CREATE TABLE OrderItems 
( 
order_num INTEGER NOT NULL, 
order_item INTEGER NOT NULL, 
prod_id CHAR(10) NOT NULL, 
quantity INTEGER NOT NULL DEFAULT 1, 
item_price DECIMAL (8,2) NOT NULL 
); 
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BY 


这 条 语句 创建 OrderItems 表 ， 包 含 构成 订单 的 各 项 ( 订单 本 身 存储 在 
Orders 表 中 )。quantity 列 为 订单 中 每 个 物品 的 数量 。 在 这 个 例子 中 ， 这 
一 列 的 描述 增加 了 DEFAULT 1, 指示 DBMS , 如 果 不 给 出 数量 则 使 用 数量 1。 









































默认 值 经 常用 于 日 期 或 时 间 戳 列 。 例 如 ， 通 过 指定 引用 系统 日 期 的 函数 
或 变量 ， 将 系统 日 期 用 作 默 认 日 期 。MySQL 用 户 指 定 DEFAULT 
CURRENT_DATE() Oracle 用 户 指 定 DEFAULT SYSDATE， 而 SQL Server 
用 户 指定 DEFAULT GETDATE() 。 遗 憾 的 是 ， 这 条 获得 系统 日 期 的 命令 在 
不 同 的 DBMS 中 几乎 都 是 不 同 的 。 表 17-1 列 出 了 这 条 命令 在 某 些 DBMS 
中 的 语法 。 这 里 知 未 列 出 某 个 DBMS ， 请 参阅 相应 的 文档 。 












































表 17-1 获得 系统 日 期 





DBMS 函数 /变量 
Access NOW() 
DB2 CURRENT_DATE 
MySQL CURRENT_DATE OO 
Oracle SYSDATE 
PostgreSQL CURRENT_DATE 
SQL Server GETDATEO 
SQLite date('now') 





提示 : 使 用 DEFAULT 而 不 是 NULL 值 
许多 数据 库 开 发 人 员 喜 欢 使 用 DEFAULT 值 而 不 是 NULL 列 ， 对 于 用 于 
计算 或 数据 分 组 的 列 更 是 如 此 。 











17.2 更 新 表 
更 新 表 定 义 ， 可 以 使 用 ALTER TABLE 语句 。 虽 然 所 有 的 DBMS 都 支持 
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ALTER TABLE， 但 它们 所 人 允许 更 新 的 内 容 差 别 很 大 。 以 下 是 使 用 ALTER 
TABLE 时 需要 考虑 的 事情 。 





口 理想 情况 下 ， 不 要 在 表 中 包含 数据 时 对 其 进行 更 新 。 应 该 在 表 的 设 
计 过 程 中 充分 考虑 未 来 可 能 的 需求 ， 避 人 免 今后 对 表 的 结构 做 大 
改动 。 

口 所 有 的 DBMS 都 允许 给 现 有 的 表 增 加 列 , 不 过 对 所 增加 列 的 数据 类 型 

( 以 及 NULL 和 DEFAULT 的 使 用 ) 有 所 限制 。 

O 许多 DBMS 不 允许 删除 或 更 改 表 中 的 列 。 

O 多 数 DBMS 允许 重新 命名 表 中 的 列 。 

O 许多 DBMS 限制 对 已 经 填 有 数据 的 列 进行 更 改 , 对 未 
乎 没有 限制 。 














T 





有 数据 的 列 几 








可 以 看 出 ， 对 已 有 表 做 更 改 既 复杂 又 不 统一 。 对 表 的 结构 能 进行 何 种 更 
改 ， 请 参阅 具体 的 DBMS 文档 。 








使 用 ALTER TABLE 更 改 表 结 构 ， 必 须 给 出 下 面 的 信息 : 


O 在 ALTER TABLE 之 后 给 出 要 更 改 的 表 名 (该 表 必 须 存 在 ， 和 否则 将 
出 错 ); 
口 列 出 要 做 哪些 更 改 。 











因为 给 已 有 表 增 加 列 可 能 是 所 有 DBMS 都 支持 的 唯一 操作 ， 所 以 我 们 举 
个 这 样 的 例子 : 


输入 了 


ALTER TABLE Vendors 
ADD vend_phone CHAR(20) ; 
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BY 


这 条 语句 给 Vendors 表 增 加 一 个 名 为 vend_phone 的 列 ， 其 数据 类 型 
为 CHAR。 





更 改 或 删除 列 、 增 加 约束 或 增加 键 , 这 些 操作 也 使 用 类 似 的 语法 ( 注意 ， 
下 面 的 例子 并 非 对 所 有 DBMS 都 有 效 ): 


输入 了 


ALTER TABLE Vendors 
DROP COLUMN vend_phone ; 


复杂 的 表 结 构 更 改 一 般 需 要 手动 删除 过 程 ， 它 涉及 以 下 步 又 : 





(1) 用 新 的 列 布局 创建 一 个 新 表 ; 

(2) 使 用 INSERT SELECT 语句 (关于 这 条 语句 的 详细 介绍 ， 请 参阅 第 15 

课 ) 从 旧 表 复制 数据 到 新 表 。 有 必要 的 话 ， 可 以 使 用 转换 函数 和 计算 
字段 ; 

(3) 检验 包含 所 需 数据 的 新 表 ; 

(4) 重 命名 旧 表 ( 如果 确 定 ， 可 以 删除 它 ); 

(5) 用 旧 表 原来 的 名 字 重 命名 新 表 ; 

(6) 根据 需要 ， 重 新 创建 触发 器 、 存 储 过 程 、 索 引 和 外 键 。 


























说 明 : ALTER TABLE 和 SOLite 

SQLite 对 使 用 ALTER TABLE 执行 的 操作 有 所 限制 。 最 重要 的 一 个 限制 
是 ， 它 不 支持 使 用 ALTER TABLE 定义 主键 和 外 键 , 这 些 必须 在 最 初创 
建 表 时 指定 。 
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注意 : 小 心 使 用 ALTER TABLE 

使 用 ALTER TABLE 要 极为 小 心 ， 应 该 在 进行 改动 前 做 完整 的 备份 ( 表 
结构 和 数据 的 备份 )。 数据 库 表 的 更 改 不 能 撤销 , 如 果 增 加 了 不 需要 的 
列 ， 也 许 无 法 删除 它们 。 类 似 地 ， 如 果 删 除了 不 应 该 删除 的 列 ， 可 能 
会 丢失 该 列 中 的 所 有 数据 。 











17.3 MRK 


删除 表 〈 删除 整个 表 而 不 
即 可 : 











其 内 容 ) 非常 简单 ， 使 用 DROP TABLE 语句 


Rm 


输入 了 


DROP TABLE CustCopy ; 


BY 





这 条 语句 删除 CustCopy K (第 15 课 中 创建 的 )。 删 除 表 没有 确认 ， 也 
不 能 撤销 ， 执 行 这 条 语句 将 永久 删除 该 表 。 





提示 : 使 用 关系 规则 防止 意外 删除 

许多 DBMS 允许 强制 实施 有 关 规 则 ， 防 止 删除 与 其 他 表 相 关联 的 表 。 在 
实施 这 些 规则 时 ， 如 果 对 某 个 表 发 布 一 条 DROP TABLE 语句 ， 且 该 表 是 
某 个 关系 的 组 成 部 分 ， 则 DBMS 将 阻止 这 条 语句 执行 ， 直 到 该 关系 被 删 
除 为 止 。 如 果 允 许 ， 应 该 启用 这 些 选项 ， 它 能 防止 意外 删除 有 用 的 表 。 











17.4 EMER 
每 个 DBMS 对 表 重 命名 的 支持 有 所 不 同 。 对 于 这 个 操作 ， 不 存在 严格 的 
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标准 。DB2、MariaDB、MySQL、Oracle 和 PostgreSQL 用 户 使 用 RENAME 
语句 ,SQL Server 用 户 使 用 sp_rename 存储 过 程 ,SQLite 用 户 使 用 ALTER 
TABLE 语句 。 

















所 有 重 命名 操作 的 基本 语法 都 要 求 指 定 旧 表 名 和 新 表 名 。 不 过 ， 存 在 
DBMS 实现 差异 。 关 于 具体 的 语法 ， 请 参阅 相应 的 DBMS 文档 。 








17.5 小结 


这 一 课 介 绍 了 几 条 新 的 SQL 语句 。CREATE TABLE 用 来 创建 新 表 , ALTER 
TABLE 用 来 更 改 表 列 (或 其 他 诸如 约束 或 索引 等 对 象 )， 而 DROP TABLE 
用 来 完整 地 删除 一 个 表 。 这 些 语句 必须 小 心 使 用 ， 并 且 应 该 在 备份 后 使 
用 。 由 于 这 些 语句 的 语法 在 不 同 的 DBMS 中 有 所 不 同 ， 所 以 更 详细 的 信 
息 请 参阅 相应 的 DBMS 文档 。 
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这 一 课 将 介绍 什么 是 视图 ， 它 们 怎样 工作 ， 何 时 使 用 它们 ; 还 将 讲述 如 
何 利 用 视图 简化 前 几 课 中 执行 的 茶 些 SQL 操作 。 























18.1 视图 


视图 是 虚拟 的 表 。 与 包含 数据 的 表 不 一 样 ， 视 图 只 包含 使 用 时 动态 检索 
数据 的 查询 。 











说 明 : DBMS 支持 
Microsoft Access 不 支持 视图 ,没有 与 SQL 视图 一 致 的 工作 方式 。 因 此 ， 
这 一 课 的 内 容 不 适用 Microsoft Access。 


MySQL 从 版 本 5 起 开始 支持 视图 ， 因此, 这 一 课 的 内 容 不 适用 较 早 版 
本 的 MySQL。 


SQLite 仅 支持 只 读 视 图 ， 所 以 视图 可 以 创建 ， 可 以 读 ， 但 其 内 容 不 能 
更 改 。 














理解 视图 的 最 好 方法 是 看 例子 。 第 12 课 用 下 面 的 SELECT 语句 从 三 个 表 
中 检索 数据 : 
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输入 了 


SELECT cust_name, cust_contact 

FROM Customers, Orders, OrderItems 

WHERE Customers.cust_id = Orders.cust_id 
AND OrderItems.order_num = Orders.order_num 
AND prod_id = 'RGANO1'; 





此 查询 用 来 检索 订购 了 某 种 产品 的 顾客 。 任 何 需 要 这 个 数据 的 人 都 必须 
理解 相关 表 的 结构 ， 知 道 如 何 创建 查询 和 对 表 进 行 联结 。 检 索 其 他 产品 
(或 多 个 产品 ) 的 相同 数据 ， 必 须 修 改 最 后 的 WHERE FA 











现在 ， 假 如 可 以 把 整个 查询 包装 成 一 个 名 为 ProductCustomers 的 虚拟 
表 ， 则 可 以 如 下 轻松 地 检索 出 相同 的 数据 : 





输入 了 


SELECT cust_name, cust_contact 
FROM ProductCustomers 
WHERE prod_id = 'RGANO1'; 

















这 就 是 视图 的 作用 。ProductCustomers 是 一 个 视图 ， 作 为 视图 ， 它 不 包 
含 任何 列 或 数据 ,包含 的 是 一 个 查询 ( 与 上 面 用 以 正确 联结 表 的 相同 查询 )。 























提示 : DBMS 的 一 致 支持 
我 们 欣慰 地 了 解 到 ， 所 有 DBMS 非常 一 致 地 支持 视图 创建 语法 。 











18.1.1 为 什么 使 用 视图 
我 们 已 经 看 到 了 视图 应 用 的 一 个 例子 。 下 面 是 视图 的 一 些 常见 应 用 。 


口 重用 SQL 语句 。 
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口 简化 复杂 的 SQL 操作 。 在 编写 查询 后 , 可 以 方便 地 重用 它 而 不 必 知 道 
REKAH 

O 使 用 表 的 一 部 分 而 不 是 整个 表 。 

O 保护 数据 。 可 以 授予 用 户 访问 表 的 特定 部 分 的 权限 ， 而 不 是 整个 表 的 
访问 权限 。 

口 更 改 数据 格式 和 表示 。 视 图 可 返回 与 底层 表 的 表示 和 格式 不 同 的 数据 。 





























创建 视图 之 后 ， 可 以 用 与 表 基 本 相同 的 方式 使 用 它们 。 可 以 对 视图 执行 
SELECT 操作 , 过 滤 和 排序 数据 , 将 视图 联结 到 其 他 视图 或 表 , 其 至 添加 
和 更 新 数据 ( 添加 和 更 新 数据 存在 某 些 限制 , 关于 这 个 内 容 稍 后 做 介绍 )。 











重要 的 是 ， 要 知道 视图 仅仅 是 用 来 查看 存储 在 别处 数据 的 一 种 设施 。 视 
图 本 身 不 包含 数据 ， 因 此 返回 的 数据 是 从 其 他 表 中 检索 出 来 的 。 在 添加 
或 更 改 这 些 表 中 的 数据 时 ， 视 图 将 返回 改变 过 的 数据 。 











注意 : 性 能 问题 

因为 视图 不 包含 数据 ， 所 以 每 次 使 用 视图 时 ， 都 必须 处 理 查询 执行 时 
A RRM Z 325 oil ge 0] # Y 3 2 604 E] X. 22 S 
套 了 视图 ， 可 能 会 下 降 得 很 厉害 。 因 此 ， 在 部 署 使 用 了 大 量 视 图 
的 应 用 前 ， 应 该 e. 











18.1.2 ”视图 的 规则 和 限制 


创建 视图 前 ， 应 该 知道 它 的 一 些 限制 。 不 过 ， 这 些 限 制 随 不 同 的 DBMS 
而 不 同 ， 因 此 在 创建 视图 时 应 该 查看 具体 的 DBMS 文档 。 
































下 面 是 关于 视图 创建 和 使 用 的 一 些 最 常见 的 规则 和 限制 。 





口 与 表 一 样 ， 视 图 必须 唯一 命名 ( 不 能 给 视图 取 与 别 的 视图 或 表 相 同 的 
名 字 )。 
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口 对 于 可 以 创建 的 视图 数目 没有 限制 。 

口 创建 视图 ， 必 须 具 有 足够 的 访问 权限 。 这 些 权 限 通 常 由 数据 库 管 理 人 

员 授 予 。 

口 视图 可 以 租 套 ， 即 可 以 利用 从 其 他 视图 中 检索 数据 的 查询 来 构造 视 

图 。 所 允许 的 般 套 层 数 在 不 同 的 DBMS 中 有 所 不 同 REWERS 
严重 降低 查询 的 性 能 ， 因 此 在 产品 环境 中 使 用 之 前 ， 应 该 对 其 进行 全 
面 测试 )。 

O 许多 DBMS 禁止 在 视图 查询 中 使 用 ORDER BY 子 句 。 

口 有 些 DBMS 要 求 对 返回 的 所 有 列 进 行 命 名 ,如 果 列 是 计算 字段 ， 则 需 

要 使 用 别名 (关于 列 别名 的 更 多 信息 ， 请 参阅 第 7 课 )。 

O 视图 不 能 索引 ， 也 不 能 有 关联 的 触发 器 或 默认 值 。 

口 有 些 DBMS 把 视图 作为 只 读 的 查询 , 这 表示 可 以 从 视图 检索 数据 , 但 

不 能 将 数据 写 回 底层 表 。 详 情 请 参阅 具体 的 DBMS 文档 。 

口 有 些 DBMS 允许 创建 这 样 的 视图 ， 它 不 能 进行 导致 行 不 再 属于 视图 的 

插入 或 更 新 。 例如 有 一 个 视图 ， 只 检索 带 有 电子 邮件 地 址 的 顾客 。 如果 

更 新 某 个 顾客 , 删除 他 的 电子 邮件 地 址 , 将 使 该 顾客 不 再 属于 视图 。 这 

是 默认 行为 , 而且 是 允许 的 , 但 有 的 DBMS 可 能 会 防止 这 种 情况 发 生 。 








































































































提示 : 参阅 具体 的 DBMS 文档 
上 面 的 规则 不 少 , 而 具体 的 DBMS 文档 很 可 能 还 包含 别 的 规则 。 因 此 ， 
在 创建 视图 前 ， 有 必要 花 点 时 间 了 解 必 须 遵 守 的 规定 。 











18.2 创建 视图 
理解 了 什么 是 视图 以 及 管理 它们 的 规则 和 约束 后 ， 我 们 来 创建 视图 。 























视图 用 CREATE VIEW 语句 来 创建 。 与 CREATE TABLE 一 样 , CREATE VIEW 
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只 能 用 于 创建 不 存在 的 视 





Psl 








说 明 : 视图 重 命名 
删除 视图 ， 可 以 使 用 DROP 语句 ， 其 语法 为 DROP VIEW viewname;。 





履 盖 〈 或 更 新 ) 视图 ， 必 须 先 删除 它 ， 然 后 再 重新 创建 。 








18.2.1 利用 视图 简化 复杂 的 联结 


一 个 最 常见 的 视图 应 用 是 隐藏 复杂 的 SQL， 这 通常 涉及 联结 。 请 看 下 面 
的 例子 : 








输入 了 


CREATE VIEW ProductCustomers AS 

SELECT cust_name, cust_contact, prod_id 

FROM Customers, Orders, OrderItems 

WHERE Customers.cust_id = Orders.cust_id 

AND OrderItems.order_num = Orders.order_num; 


BY 





这 条 语句 创建 一 个 名 为 ProductCustomers 的 视图 ， 它 联结 三 个 表 ， 返 
回 已 订购 了 任意 产品 的 所 有 顾客 的 列表 。 如 果 执 行 SELECT * FROM 
ProductCustomers， 将 列 出 订购 了 任意 产品 的 顾客 。 








仿 索 订购 了 产品 RGAN01 的 顾客 ， 可 如 下 进行 : 
输入 了 


SELECT cust_name, cust_contact 


FROM ProductCustomers 
WHERE prod_id = 'RGANO1'; 
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输出 

cust_name cust_contact 
et Dn o Stephene 
The Toy Store Kim Howard 

SH Y 


这 条 语句 通过 WHERE 子 句 从 视图 中 检索 特定 数据 。 当 DBMS 处 理 此 查询 
时 , 它 将 指定 的 WHERE 子 句 添 加 到 视图 查询 中 已 有 的 WHERE 子 句 中 ， 以 
便 正 确 过 滤 数 据 。 


可 以 看 出 ， 视 图 极 大 地 简化 了 复杂 SQL 语句 的 使 用 。 利 用 视图 ， 可 一 次 
性 编写 基础 的 SQL， 然后 根据 需要 多 次 使 用 。 



































提示 : 创建 可 重用 的 视图 
创建 不 绑 定 特定 数据 的 视图 是 一 种 好 办 法 。 例 如 ， 上 面 创 建 的 视图 返 
回 订 购 所 有 产品 而 不 仅仅 是 RGAN01 的 顾客 ( 这 个 视图 先 创 建 )。 扩展 
视图 的 范围 不 仅 使 得 它 能 被 重用 ， 而 且 可 能 更 有 用 。 这 样 做 不 需要 创 
建 和 维护 多 个 类 似 视图 。 











18.2.2 ”用 视图 重新 格式 化 检索 出 的 数据 


如 前 所 述 ， 视 图 的 另 一 常见 用 途 是 重新 格式 化 检索 出 的 数据 。 下 面 的 
SELECT 语句 〈 来 自 第 7 课 ) 在 单个 组 合计 算 列 中 返回 供应 商 名 和 位 置 : 





输入 了 


SELECT RTRIM(vend_name) + ' (' + RTRIM(vend_country) + ')' 
AS vend_title 

FROM Vendors 

ORDER BY vend_name; 
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输出 了 


vend_title 

Bear Emporium (USA) 
Bears R Us (USA) 

Doll House Inc. (USA) 
Fun and Games (England) 
Furball Inc. (USA) 
Jouets et ours (France) 


下 面 是 相同 的 语句 ， 但 使 用 了 11 语 法 〈 如 第 7 课 所 述 ): 
输入 了 


SELECT RTRIM(vend_name) || ' (' || RTRIM(vend_country) || ')' 
AS vend_title 

FROM Vendors 

ORDER BY vend_name; 




















输出 


vend_title 

Bear Emporium (USA) 
Bears R Us (USA) 

Doll House Inc. (USA) 
Fun and Games (England) 
Furball Inc. (USA) 
Jouets et ours (France) 


现在 , 假设 经 常 需要 这 个 格式 的 结果 。 我 们 不 必 在 每 次 需要 时 执行 这 种 拼接 ， 
而 是 创建 一 个 视图 ， 使 用 它 即 可 。 把 此 语句 转换 为 视图 ， 可 按 如 下 进行 : 











输入 了 


CREATE VIEW VendorLocations AS 

SELECT RTRIM(vend_name) + ' (' + RTRIM(vend_country) + ')' 
AS vend_title 

FROM Vendors; 
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下 面 是 使 用 |1 语 法 的 相同 语句 : 


输入 
CREATE VIEW VendorLocations AS 
SELECT RTRIM(vend_name) || ' (' || RTRIM(vend_country) || ')' 


AS vend_title 
FROM Vendors; 


BY 





这 条 语句 使 用 与 以 前 SELECT 语句 相同 的 查询 创建 视图 。 要 检索 数据 ， 
创建 所 有 的 邮件 标签 ， 可 如 下 进行 : 








输入 了 


SELECT * 
FROM VendorLocations ; 


输出 了 


vend_title 

Bear Emporium (USA) 
Bears R Us (USA) 

Doll House Inc. (USA) 
Fun and Games (England) 
Furball Inc. (USA) 
Jouets et ours (France) 





说 明 : SELECT 约束 全 部 适用 

在 这 一 课 的 前 面 提 到 ， 各 种 DBMS 中 用 来 创建 视图 的 语法 相当 一 致 。 
那么 ， 为 什么 会 有 多 种 创建 视图 的 语句 版 本 呢 ? 因为 视图 只 包含 一 个 
SELECT 语句 ， 而 这 个 语句 的 语法 必须 遵循 具体 DBMS 的 所 有 规则 和 
约束 ， 所 以 会 有 多 个 创建 视图 的 语句 版 本 。 
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18.2.3 ”用 视图 过 滤 不 想 要 的 数据 


视图 对 于 应 月 








普通 的 WHERE 子 句 也 很 有 用 。 例 如 ， 可 以 定义 





CustomerEMailList 视图 , 过 滤 没 有 电子 邮件 地 址 的 顾客 。 为 此 , 可 使 


用 下 面 的 语句 : 


输入 了 


CREATE VIEW CustomerEMailList AS 
SELECT cust_id, cust_name, cust_email 


FROM Customers 
WHERE cust_email IS NOT NULL; 


BY 





显然 ， 在 将 电子 邮件 发 送 到 邮件 列表 时 ， 需 要 排除 没有 电子 邮件 地 址 的 
用 户 。 这 里 的 WHERE 子 句 过 滤 了 cust_email 列 中 具有 NULL 值 的 那些 
行 ， 使 它们 不 被 检索 出 来 。 


现在 ， 可 以 像 使 用 其 他 表 一 样 使 月 


输入 了 


SELECT * 





FROM CustomerEMailList; 


1000000001 
1000000003 
1000000004 


cust_name 
Village Toys 
Fun4A11 
Fun4A11 
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日 视图 CustomerEMai1List。 


cust_email 


sales@villagetoys.com 
jjonesQfun4a11.com 
dstephens@fun4all.com 
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说 明 : WHERE 子 句 与 WHERE F4 
从 视图 检索 数据 时 如 果 使 用 了 一 条 WHERE 子 句 ， 则 两 组 子 句 (一 组 在 
视图 中 ， 另 一 组 是 传递 给 视图 的 ) 将 自动 组 合 。 








18.24 使 用 视图 与 计算 字段 


在 简化 计算 字段 的 使 用 上 ， 视 图 也 特别 有 用 。 下 面 是 第 7 课 中 介绍 的 一 
条 SELECT 语句 ， 它 检索 某 个 订单 中 的 物品 ， 计 算 每 种 物品 的 总 价格 : 























输入 了 


SELECT prod_id， 

quantity, 

item_price, 

quantity*item_price AS expanded_price 
FROM OrderItems 
WHERE order_num = 20008; 


输出 了 

prod_id quantity item_price expanded_price 
RGAN01 5 4.9900 24.9500 

BRO3 5 11.9900 59.9500 

BNBG01 10 3.4900 34.9000 

BNBG02 10 3.4900 34.9000 

BNBG03 10 3.4900 34.9000 


要 将 其 转换 为 一 个 视图 ， 如 下 进行 : 
输入 了 


CREATE VIEW OrderItemsExpanded AS 
SELECT order_num, 

prod_id， 

quantity, 
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item_price， 
quantity*item_price AS expanded_price 
FROM OrderItems ; 


检索 订单 20008 的 详细 内 容 ( 上面 的 输出 )， 如 下 进行 : 


输入 了 


SELECT * 
FROM OrderItemsExpanded 
WHERE order_num = 20008; 


输出 了 

order_num prod_id quantity item_price expanded_price 
20008 RGANO1 5 4.99 24.95 
20008 BRO3 5 11.99 59.95 
20008 BNBGO1 10 3.49 34.90 
20008 BNBGO2 10 3.49 34.90 
20008 BNBGO3 10 3.49 34.90 





可 以 看 到 ,视图 非常 容易 创建 ， 而 且 很 好 使 用 。 正 确 使 用 ， 视 图 可 极 大 
地 简化 复杂 数据 的 处 理 。 





18.3 小结 


视图 为 虚拟 的 表 。 它 们 包含 的 不 是 数据 而 是 根据 需要 检索 数据 的 查询 。 
视图 提供 了 一 种 封装 SELECT 语句 的 层次 ， 可 用 来 简化 数据 处 理 ， 重 新 
格式 化 或 保护 基础 数据 。 
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这 一 课 介绍 什么 是 存储 过 程 ， 为 什么 要 使 用 存储 过 程 ， 如 何 使 用 存储 过 
程 ， 以 及 创建 和 使 用 存储 过 程 的 基本 语法 。 








19.1 存储 过 程 


今 为 止 , 我 们 使 用 的 大 多 数 jja 语句 都 是 针对 一 个 或 多 个 表 的 单条 语 
= 并 非 所 有 操作 都 这 么 简单 ， 经 常会 有 一 些 复杂 的 操作 需要 多 条 语句 
才能 完成 ， 例 如 以 下 的 情形 。 


O 为 了 处 理 订单 ， 必 须 核对 以 保证 库存 中 有 相应 的 物品 。 

口 如 果 物 品 有 库存 ， 需 要 预定 ， 不 再 出 售 给 别 的 人 ， 并 且 减 少 物品 数据 
以 反映 正确 的 库存 量 。 

口 库存 中 没有 的 物品 需要 订购 ， 这 需要 与 供应 商 进行 某 种 交互 。 

口 关于 哪些 物品 和 人 库 (并 且 可 以 立即 发 货 ) 和 哪些 物品 退 订 ， 需 要 通知 
相应 的 顾客 。 















































这 显然 不 是 一 个 完整 的 例子 ， 它 甚至 超出 了 本 书 中 所 用 样 例 表 的 范围 ， 
但 足以 表达 我 们 的 意思 了 。 执行 这 个 处 理 需 要 针对 许多 表 的 多 条 SQL 语 
句 。 此 外 ， 需 要 执行 的 具体 SQL 语句 及 其 次 序 也 不 是 固定 的 ， 它 们 可 能 
会 根据 物品 是 否 在 库存 中 而 变化 
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那么 ， 怎 样 编写 代码 呢 ? 可 以 单独 编写 每 条 SQL 语句 ， 并 根据 结果 有 条 
件 地 执行 其 他 语句 ,在 每 次 需要 这 个 处 理 时 ( 以 及 每 个 需要 它 的 应 用 中 ), 
都 必须 做 这 些 工 作 。 























可 以 创建 存储 过 程 。 简 单 来 说 ， 存 储 过 程 就 是 为 以 后 使 用 而 保存 的 一 条 
或 多 条 SQL 语句 。 可 将 其 视 为 批文 件 , 虽然 它们 的 作用 不 仅 限于 批 处 理 。 

















说 明 : 具体 DBMS 的 支持 

Microsoft Access 和 SQLite 不 支持 存储 过 程 。 因 此 , 本 课 的 内 容 不 适用 
GAT 

MySQL 5 已 经 支持 存储 过 程 。 因 此 ， 本 课 的 内 容 不 适用 MySQL 较 早 
的 版 本 。 








说 明 : 还 有 更 多 内 容 

存储 过 程 很 复杂 ， 全 面 介绍 它 需 要 很 大 篇 幅 。 本 课 不 打算 讲解 存储 过 
程 的 所 有 内 容 ， 只 给 出 简单 介绍 ， 让 读者 对 它们 的 功能 有 所 了 解 。 因 
此 ， 这 里 给 出 的 例子 只 提供 Oracle 和 SQL Server 的 语法 。 











19.2 为 什么 要 使 用 存储 过 程 


我 们 知道 了 什么 是 存储 过 程 ， 那 么 为 什么 要 使 用 它们 呢 ? 理由 很 多 ， 下 
面 列 出 一 些 主要 的 。 
































O 通过 把 处 理 封装 在 一 个 易 用 的 单元 中 ， 可 以 简化 复杂 的 操作 ( 如 前 面 
例子 所 述 )。 

口 由 于 不 要 求 反复 建立 一 系列 处 理 步 又 ， 因 而 保证 了 数据 的 一 致 性 。 如 
果 所 有 开发 人 员 和 应 用 程序 都 使 用 同一 存储 过 程 ， 则 所 使 用 的 代码 都 
是 相同 的 。 
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这 一 点 的 延伸 就 是 防止 错误 。 需 要 执行 的 步骤 越 多 ， 出 错 的 可 能 性 就 
越 大 。 防 止 错误 保证 了 数据 的 一 致 性 。 

口 简化 对 变动 的 管理 。 如 果 表 名 、 列 名 或 业务 逻辑 ( 或 别 的 内 容 ) 有 变 
化 ， 那 么 只 需要 更 改 存储 过 程 的 代码 。 使 用 它 的 人 员 黄 至 不 需要 知道 
这 些 变化 。 
这 一 点 的 延伸 就 是 安全 性 。 通 过 存储 过 程 限 制 对 基础 数据 的 访问 ， 减 
少 了 数据 论 误 〈 无 意识 的 或 别 的 原因 所 导致 的 数据 泡 误 ) 的 机 会 。 

口 因为 存储 过 程 通常 以 编译 过 的 形式 存储 , 所 以 DBMS 处 理 命令 所 需 的 

工作 量 少 ， 提 高 了 性 能 。 

O 存在 一 些 只 能 用 在 单个 请 求 中 的 SQL 元 素 和 特性 , 存储 过 程 可 以 使 用 
它们 来 编写 功能 更 强 更 灵活 的 代码 。 



























































换 句 话说 ,使 用 存储 过 程 有 三 个 主要 的 好 处 ， 即 简单 、 安 全 、 高 性 能 。 
显然 , 它们 都 很 重要 。 不 过 ， 在 将 SQL 代码 转换 为 存储 过 程 前 ， 也 必须 
知道 它 的 一 些 缺 陷 。 








O 不 同 DBMS 中 的 存储 过 程 语法 有 所 不 同 。 事 实 上 ,编写 真正 的 可 移植 
存储 过 程 几乎 是 不 可 能 的 。 不 过 ， 存 储 过 程 的 自我 调用 〈 名 字 以 及 数 
据 如 何 传递 ) 可 以 相对 保持 可 移植 ,因此 ,如 果 需 要 移植 到 别 的 DBMS， 
至 少 客户 端 应 用 代码 不 需要 变动 。 

O 一 般 来 说 ,编写 存储 过 程 比 编写 基本 SQL 语句 复杂 , 需要 更 高 的 技能 ， 
更 丰富 的 经 验 。 因 此 ， 许 多 数据 库 管理 员 把 限制 存储 过 程 的 创建 作为 
安全 措施 ( 主要 受 上 一 条 缺陷 的 影响 )。 















































尽管 有 这 些 缺 隐 ， 存 储 过 程 还 是 非常 有 用 的 ， 并 且 应 该 合用。 事实 上 ， 
多 数 DBMS 都 带 有 用 于 管理 数据 库 和 表 的 各 种 存储 过 程 。 更 多 信息 请 参 
阅 具体 的 DBMS 文档 。 
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说 明 : 不 能 编写 存储 过 程 ? 你 依然 可 以 使 用 

大 多 数 DBMS 将 编写 存储 过 程 所 需 的 安全 和 访问 权限 与 执行 存储 过 
程 所 需 的 安全 和 访问 权限 区 分 开 来 。 这 是 好 事情 ， 即 使 你 不 能 〈 或 
不 想 ) 编写 自己 的 存储 过 程 ， 也 仍然 可 以 在 适当 的 时 候 执 行 别 的 存 
储 过 程 。 











19.3 ”执行 存储 过 程 

存储 过 程 的 执行 远 比 编写 要 频繁 得 多 , 因此 我 们 先 介 绍 存储 过 程 的 执行 。 
执行 存储 过 程 的 SQL 语句 很 简单 ， 即 EXECUTE。EXECUTE 接受 存储 过 程 
名 和 需要 传递 给 它 的 任何 参数 。 请 看 下 面 的 例子 : 





输入 了 


EXECUTE AddNewProduct( 'JTS01', 

'Stuffed Eiffel Tower', 

6.49, 

'Plush stuffed toy with the text La 
wTour Eiffel in red white and blue' ); 


BY 


这 里 执行 一 个 名 为 AddNewProduct 的 存储 过 程 ， 将 一 个 新 产品 添加 到 
Products 表 中 。AddNewProduct 有 四 个 参数 ， 分 别 是 : 供应 商 ID 
( Vendors 表 的 主键 )、 产 品名 、 价 格 和 描述 。 这 4 个 参数 匹配 存储 过 程 
中 4 个 预期 变量 ( 定义 为 存储 过 程 自身 的 组 成 部 分 )。 此 存储 过 程 将 新 行 
添加 到 Products 表 ， 并 将 传人 的 属性 赋 给 相应 的 列 。 




















我 们 注意 到 ， 在 Products 表 中 还 有 另 一 个 需要 值 的 列 prod_id 列 ， 它 
是 这 个 表 的 主键 。 为 什么 这 个 值 不 作为 属性 传递 给 存储 过 程 ? 要 保证 恰 
当地 生成 此 ID ， 最 好 是 使 生成 此 ID 的 过 程 自动 化 ( 而 不 是 依赖 于 最 终 








ce 
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用 户 的 输入 )。 这 也 是 这 个 例子 使 用 存储 过 程 的 原因 。 以 下 是 存储 过 程 所 
完成 的 工作 : 








O 验证 传递 的 数据 ， 保 证 所 有 4 个 参数 都 有 值 ; 

口 生成 用 作 主 键 的 唯一 D; 

O 将 新 产品 插入 Products 表 ， 在 合适 的 列 中 存储 生成 的 主键 和 传递 的 
数据 。 












































这 就 是 存储 过 程 执行 的 基本 形式 。 对 于 有 具体 的 DBMS ， 可 能 包括 以 下 的 
执行 选择 : 


口 参数 可 选 ， 具 有 不 提供 参数 时 的 默认 值 ; 

口 不 按 次 序 给 出 参数 ， 以 “参数 = 值 ”的 方式 给 出 参数 值 。 

口 输出 参数 ， 人 允许 存储 过 程 在 正 执行 的 应 用 程序 中 更 新 所 用 的 参数 。 
口 用 SELECT 语句 检索 数据 。 

口 返回 代码 ， 人 允许 存储 过 程 返回 一 个 值 到 正在 执行 的 应 用 程序 。 























19.4 创建 存储 过 程 


正如 所 述 ,存储 过 程 的 编写 很 重要 。 为 了 获得 感性 认识 ， 我 们 来 看 一 个 简 
单 的 存储 过 程 例子 ， 它 对 邮件 发 送 清单 中 具有 邮件 地 址 的 顾客 进行 计数 。 























下 面 是 该 过 程 的 Oracle 版 本 : 





输入 了 


CREATE PROCEDURE MailingListCount ( 
ListCount OUT INTEGER 

) 

IS 

v_rows INTEGER; 
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BEGIN 
SELECT COUNT(*) INTO v_rows 
FROM Customers 
WHERE NOT cust_email IS NULL; 
ListCount := v_rows; 

END; 


BY 


这 个 存储 过 程 有 一 个 名 为 ListCount 的 参数 。 此 参数 从 存储 过 程 返回 一 
个 值 而 不 是 传递 一 个 值 给 存储 过 程 。 关 键 字 OUT 用 来 指示 这 种 行为 。 
Oracle 支持 IN 传递 值 给 存储 过 程 ) OUT( 从 存储 过 程 返回 值 , 如 这 里 )、 
INOUT ( 既 传 递 值 给 存储 过 程 也 从 存储 过 程 传 回 值 ) 类 型 的 参数 。 存 储 
过 程 的 代码 括 在 BEGIN 和 END 语句 中 ， 这 里 执行 一 条 简单 的 SELECT 语 
句 ， 它 检索 具有 邮件 地 址 的 顾客 。 然 后 用 检索 出 的 行 数 设置 ListCount 
(要 传递 的 输出 参数 )。 



































调用 Oracle 例子 可 以 像 下 面 这 样 : 
输入 


var ReturnValue NUMBER 


EXEC MailingListCount(:ReturnValue); 
SELECT ReturnValue; 


BY 


这 段 代 码 声 明了 一 个 变量 来 保存 存储 过 程 返回 的 任何 值 ， 然 后 执行 存储 
过 程 ， 再 使 用 SELECT 语句 显示 返回 的 值 。 


下 面 是 该 过 程 的 SQL Server 版 本 。 
iN Y 


CREATE PROCEDURE MailingListCount 
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AS 

DECLARE @cnt INTEGER 

SELECT @cnt = COUNT(*) 

FROM Customers 

WHERE NOT cust_email IS NULL; 
RETURN Qcnt; 


BY 


此 存储 过 程 没 有 参数 。 调 用 程序 检索 SQL Server 的 返回 代码 提供 "s 
其 中 用 DECLARE 语句 声明 了 一 个 名 为 ecnt 的 局 部 变量 ( SQL Server 中 所 
有 局 部 变量 名 都 以 @ 起 头 ); 然后 在 SELECT 语句 中 使 用 这 个 变量 ， 让 它 
包含 COUNT O 函数 返回 的 值 ; 最 后 , 用 RETURN @cnt 语句 将 计数 返回 给 
调用 程序 。 


调用 SQL Server 例子 可 以 像 下 面 这 样 : 
输入 了 


DECLARE @ReturnValue INT 
EXECUTE @ReturnValue=MailingListCount; 
SELECT @ReturnValue; 




















BY 


这 段 代 码 声 明了 一 个 变量 来 保存 存储 过 程 返回 的 任何 值 ， 然 后 执行 存储 
过 程 ， 再 使 用 SELECT 语句 显示 返回 的 值 。 

















下 面 是 另 一 个 例子 ， 这 次 在 Orders 表 中 搬 和 人 一 个 新 订单 。 此 程序 仅 适 
用 于 SQL Server， 但 它 说 明了 存储 过 程 的 某 些 用 途 和 技术 : 


输入 了 


CREATE PROCEDURE NewOrder @cust_id CHAR(10) 
AS 
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E 


-- Declare variable for order number 
DECLARE @order_num INTEGER 

-- Get current highest order number 
SELECT Qorder_num=MAX(order_num) 

FROM Orders 

-- Determine next order number 

SELECT @order_num=@order_num+1 

-- Insert new order 

INSERT INTO Orders(order_num, order_date, cust_id) 
VALUES (@order_num, GETDATE(), @cust_id) 
-- Return order number 

RETURN @order_num; 


BY 


此 存储 过 程 在 Orders 表 中 创建 一 个 新 订单 。 它 只 有 一 个 参数 ， 即 下 订 
单 顾 客 的 ID 。 订 单 号 和 订单 日 期 这 两 列 在 存储 过 程 中 自动 生成 。 代 码 首 
先 声明 一 个 局 部 变量 来 存储 订单 号 。 接 着 ,检索 当 前 最 大 订单 号 (使 用 
MAXO 函数 ) 并 增加 1 (使 用 SELECT 语句 )。 然后 用 INSERT 语句 搬入 由 
新 生成 的 订单 号 、 当 前 系统 日 期 (用 GETDATEO 函数 检索 ) 和 传递 的 顾 
客 ID 组 成 的 订单 。 最 后 ， 用 RETURN @order_num 返回 订单 号 (处理 订 
单 物品 需要 它 )。 请 注意 ,此 代码 加 了 注释 , 在 编写 存储 过 程 时 应 该 多 加 
注释 。 





















































说 明 : 注释 代码 

应 该 注释 所 有 代码 ， 存 储 过 程 也 不 例外 。 增 加 注释 不 影响 性 能 ， 因 此 
不 存在 缺陷 (除了 增加 编写 时 间 外 ) 注释 代码 的 好 处 很 多 ， 包 括 使 别 
人 (以 及 你 自己 ) 更 容易 地 理解 和 更 安全 地 修改 代码 。 

对 代码 进行 注释 的 标准 方式 是 在 之 前 放置 -- (两 个 连 字 符 )。 有 的 
DBMS 还 支持 其 他 的 注释 语法 , 不 过 所 有 DBMS 都 支持 --， 因 此 在 注 
释 代 码 时 最 好 都 使 用 这 种 语法 。 
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下 面 是 相同 SQL Server 代码 的 一 个 很 不 同 的 版 本 : 
输入 可 


CREATE PROCEDURE NewOrder @cust_id CHAR(10) 
AS 

-- Insert new order 

INSERT INTO Orders(cust_id) 

VALUES (@cust_id) 

-- Return order number 

SELECT order_num = Q@QIDENTITY; 


3 Y 


此 存储 过 程 也 在 Orders 表 中 创建 一 个 新 订单 。 这 次 由 DBMS 生成 订单 号 。 
大 多 数 DBMS 都 支持 这 种 功能 ;SQL Server 中 称 这 些 自动 增 量 的 列 为 标识 
字段 ( identity field ), 而 其 他 DBMS 称 之 为 自动 编号 (auto number ) 或 序列 
(sequence )。 传 递 给 此 过 程 的 参数 也 是 一 个 ， 即 下 订单 的 顾客 ID。 订 单 号 
和 订单 日 期 没有 给 出 , DBMS 对 日 期 使 用 默认 值 ( GETDATE Kt), 订单 
号 自动 生成 。 怎样 才 能 得 到 这 个 自动 生成 的 ID? 在 SQL Server 上 可 在 全 后 
变量 @@IDENTITY 中 得 到 ， 它 返回 到 调用 程序 ( 这 里 使 用 SELECT 语句 )。 




































































可 以 看 到 ， 借 助 存储 过 程 ， 可 以 有 多 种 方法 完成 相同 的 工作 。 不 过 ， 所 
选择 的 方法 受 所 用 DBMS 特性 的 制约 。 





19.5 ”小结 


这 一 课 介绍 了 什么 是 存储 过 程 ， 为 什么 使 用 存储 过 程 。 我 们 介绍 了 执行 
和 创建 存储 过 程 的 语法 ， 使 用 存储 过 程 的 一 些 方 法 。 存 储 过 程 是 个 相当 
重要 的 主题 ,一 课 内 容 无 法 全 部 涉及 。 各 种 DBMS 对 存储 过 程 的 实现 不 
一 ， 你 使 用 的 DBMS 可 能 提供 了 一 些 这 里 提 到 的 功能 ,也 有 其 他 未 提 及 
的 功能 ,更 详细 的 介绍 请 参阅 具体 的 DBMS 文档 。 
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这 一 课 介 绍 什么 是 


务 处 理 。 





TAARE, 如 何 利用 COMMIT 和 ROLLBACK 语句 管理 事 





201 事务 处 理 











使 用 事务 处 理 ( transaction processing )， 通 过 确保 成 批 的 SQL 操作 要 么 





完全 执行 ， 要么 完全 不 执行 ， 来 维护 数据 库 的 完整 性 。 














正如 第 12 课 所 述 , 关系 数据 库 把 数据 存储 在 多 个 表 中 , 使 数据 更 容易 操 

















纵 、 维 护 和 重用 。 不 用 深究 如 何以 及 为 什么 进行 关系 数据 库 设计 ， 在 某 
种 程度 上 说 ， 设 计 良 好 的 数据 库 模 式 都 是 关联 的 。 























前 面 使 用 的 Orders 表 就 是 一 个 很 好 的 例子 。 订 单 存储 在 Orders 和 
OrderItems 两 个 表 中 : Orders 存储 实际 的 订单 ，0rderItems 存储 订 
购 的 各 项 物品 。 这 两 个 表 使 用 称 为 主键 (参阅 第 1 课 ) 的 唯一 ID 互相 关 














联 ， 又 与 包含 客户 和 产品 信息 的 其 他 表 相 关联 。 


给 系统 添加 订单 的 过 程 如 下 : 





(1) 检查 数据 库 中 是 否 存在 相应 的 顾客 ， 如 果 不 存在 ， 添 加 他 ; 


(2) 检索 顾客 的 ID ; 
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(3) 在 Orders 表 添 加 一 行 ， 它 与 顾客 ID 相关 联 ; 

(4) 检索 Orders 表 中 赋予 的 新 订单 ID; 

(5) 为 订购 的 每 个 物品 在 OrderItems 表 中 添加 一 行 ,通过 检索 出 来 的 ID 
把 它 与 Orders 表 关 联 (并 且 通 过 产品 ID 与 Products 表 关 联 )。 























现在 假设 由 于 某 种 数据 库 故障 ( 如 超出 磁盘 空间 、 安 全 限制 、 表 锁 等 )， 
这 个 过 程 无 法 完成 。 数 据 库 中 的 数据 会 出 现 什 么 情况 ? 





如 果 故 障 发 生 在 添加 顾客 之 后 ， 添 加 Orders 表 之 前 ， 则 不 会 有 什么 问 
题 。 某 些 顾 客 没 有 订单 是 完全 合法 的 。 重 新 执行 此 过 程 时 ， 所 插入 的 顾 
客 记录 将 被 检索 和 使 用 。 可 以 有 效 地 从 出 故障 的 地 方 开始 执行 此 过 程 。 

















但 是 ， 如 果 故 障 发 生 在 插入 Orders 行 之 后 ， 添 加 OrderItems 行 之 前 ， 
怎么 办 ? 现在， 数据 库 中 有 一 个 空 订单 。 























更 糟 的 是 ， 如 果 系 统 在 添加 OrderItems 行 之 时 出 现 故 障 ， 怎 么 办 ? 结 
果 是 数据 库 中 存在 不 完整 的 订单 ， 而 你 还 不 知道 。 


























如 何 解 决 这 种 问题 ? 这 就 需要 使 用 事务 处 理 了 。 事 务 处 理 是 一 种 机 制 ， 
用 来 管理 必须 成 批 执行 的 SQL 操作 , 保证 数据 库 不 包含 不 完整 的 操作 结 
果 。 利 用 事务 处 理 ， 可 以 保证 一 组 操作 不 会 中 途 停止 ， 它 们 要 么 完全 执 
ÍT, 要 么 完全 不 执行 ( 除非 明确 指示 )。 如 果 没 有 错误 发 生 ， 整 组 语句 提 
交 给 ( 写 到 ) 数据 库 表 ; 如 果 发 生 错误 ， 则 进行 回 退 ( 撤销 ), 将 数据 库 
恢复 到 某 个 已 知 且 安 全 的 状态 。 



































再 看 这 个 例子 ， 这 次 我 们 说 明 这 一 过 程 是 如 何 工作 的 : 





(1) 检查 数据 库 中 是 否 存在 相应 的 顾客 ， 如 果 不 存在 ， 添 加 他 ; 
(2) 提交 顾客 信息 ; 
(3) 检索 顾客 的 ID 
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(4) Æ Orders 表 中 添加 一 行 ; 

(5) 如 果 向 Orders 表 添 加 行 时 出 现 故障 ， 回 退 ; 

(6) 检索 Orders 表 中 赋予 的 新 订单 ID; 

(7) 对 于 订购 的 每 项 物品 ， 添 加 新 行 到 OrderItems 表 ; 

(8) 如 果 向 OrderItems 添加 行 时 出 现 故障 , 回 退 所 有 添加 的 OrderItems 
行 和 Orders 行 。 












































在 使 用 事务 处 理 时 ， 有 几 个 反复 出 现 的 关键 词 。 下 面 是 关于 事务 处 理 需 
要 知道 的 几 个 术语 : 








FJ (transaction ) 指 一 组 SQL 语句 ; 

回 退 (rollback ) 指 撤销 指定 SQL 语句 的 过 程 ; 
提交 (commit ) 指 将 未 存储 的 SQL 语句 结果 写 人 数据 库 表 ; 

保留 点 〈savepoint ) 指 事务 处 理 中 设置 的 临时 占 位 符 ( placeholder ), 
可 以 对 它 发 布 回 退 (与 回 退 整个 事务 处 理 不 同 )。 











Ü DQ D D 

















提示 : 可 以 回 退 哪 些 语句 ? 

事务 处 理 用 来 管理 INSERT、UPDATE 和 DELETE 语句 。 不 能 回 退 SELECT 
语句 ( 回 退 SELECT 语句 也 没有 必要 )， 也 不 能 回 退 CREATE 或 DROP 操 
作 。 事务 处 理 中 可 以 使 用 这 些 语句 , 但 进行 回 退 时 , 这 些 操作 也 不 撤销 。 








20.22 ”控制 事务 处 理 
我 们 已 经 知道 了 什么 是 事务 处 理 ， 下 面 讨论 管理 事务 中 涉及 的 问题 。 

















注意 : 事务 处 理 实现 的 差异 
不 同 DBMS 用 来 实现 事务 处 理 的 语法 有 所 不 同 。 在 使 用 事务 处 理 时 请 
参阅 相应 的 DBMS 文档 。 
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理事 务 的 关键 在 于 将 SQL 语句 组 分 解 为 逻辑 块 , 并 明确 规定 数据 何 时 
该 回 退 ， 何 时 不 应 该 回 退 。 





Er m 


有 的 DBMS 要 求 明确 标识 事务 处 理 块 的 开始 和 结束 。 如 在 SQL Server 中 ， 
标识 如 下 : 


输入 了 
BEGIN TRANSACTION 


COMMIT TRANSACTION 


BY 


在 这 个 例子 中 ，BEGIN TRANSACTION 和 COMMIT TRANSACTION 语句 之 
间 的 SQL 必须 完全 执行 或 者 完全 不 执行 。 


MariaDB 和 MySQL 中 等 同 的 代码 为 : 
输入 了 


START TRANSACTION 








Oracle 使 用 的 语法 : 


输入 了 


SET TRANSACTION 





PostgreSQL 使 用 ANSI SQL 语法 : 
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输入 了 


BEGIN 


其 他 DBMS 采用 上 述 语法 的 变 体 。 你 会 发 现 ， 多 数 实现 没有 明确 标识 事 
务 处 理 在 何 处 结束 。 事 务 一 直 存 在 ， 直 到 被 中 断 。 通 常 ，COMMITT 用 于 
保存 更 改 ，ROLLBACK 用 于 撤销 ， 详 述 如 下 。 











20.2.1 使 用 ROLLBACK 











SQL 的 ROLLBACK 命令 用 来 回 退 (撤销 ) SQL 语句 ， 请 看 下 面 的 语句 : 


输入 

DELETE FROM Orders; 
ROLLBACK; 

S Y 


在 此 例子 中 , 执行 DELETE 操作 , 然后 用 ROLLBACK 语句 撤销 。 虽然 这 不 
是 最 有 用 的 例子 ,但 它 的 确 能 够 说 明 , 在 事务 处 理 块 中 , DELETE 操作 (与 
INSERT 和 UPDATE 操作 一 样 ) 并 不 是 最 终 的 结果 。 








20.2.2 ”使 用 COMMIT 








一 般 的 SQL 语句 都 是 针对 数据 库 表 直接 执行 和 编写 的 。 这 就 是 所 请 的 隐 
式 提交 (implicit commit )， 即 提交 ( 写 或 保存 ) 操作 是 自动 进行 的 。 


























在 事务 处 理 块 中 ,提交 不 会 隐 式 进行 。 不 过 ,不同 DBMS 的 做 法 有 所 不 
同 。 有 的 DBMS 按 隐 式 提交 处 理事 务 端 ， 有 的 则 不 这 样 。 
































进行 明确 的 提交 ， 使 用 COMMIT 语句 。 下 面 是 一 个 SQL Server 的 例子 : 
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输入 了 


BEGIN TRANSACTION 

DELETE OrderItems WHERE order_num = 12345 
DELETE Orders WHERE order_num = 12345 
COMMIT TRANSACTION 


BY 


在 这 个 SQL Server 例子 中 ， 从 系统 中 完全 删除 订单 12345。 因 为 涉及 更 
新 两 个 数据 库 表 Orders 和 0rderItems, 所 以 使 用 事务 处 理 块 来 保证 订 
单 不 被 部 分 删除 。 最 后 的 COMMIT 语句 仅 在 不 出 错时 写 出 更 改 。 如 果 第 
一 条 DELETE 起 作用 ,但 第 二 条 失败 ， 则 DELETE 不 会 提交 。 











为 在 Oracle 中 完成 相同 的 工作 ， 可 如 下 进行 : 
输入 了 


SET TRANSACTION 

DELETE OrderItems WHERE order_num = 12345; 
DELETE Orders WHERE order_num = 12345; 
COMMIT; 


20.2.3 ”使 用 保留 点 


使 用 简单 的 ROLLBACK 和 COMMIT 语句 , 就 可 以 写 入 或 撤销 整个 事务 。 但 
E, 只 对 简单 的 事务 才能 这 样 做 , 复杂 的 事务 可 能 需要 部 分 提交 或 回 退 。 























例如 前 面 描述 的 添加 订单 的 过 程 就 是 一 个 事务 。 如 果 发 生 错 误 ， 只 需要 
返回 到 添加 Orders 行 之 前 即 可 。 不 需要 回 退 到 Customers 表 ( 如 果 存 
在 的 话 )。 














要 支持 回 退 部 分 事务 ， 必 须 在 事务 处 理 块 中 的 合适 位 置 放置 占 位 符 。 这 
样 ， 如 果 需 要 回 退 ， 可 以 回 退 到 某 个 占 位 符 。 
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在 SQL 中 ， 这 些 占 位 符 称 为 保留 点 。 在 MariaDB, MySQL 和 Oracle 中 
创建 占 位 符 ， 可 使 用 SAVEPOINT 语句 。 


iW A Y 

SAVEPOINT deletel; 

在 SQL Server 中 ， 如 下 进行 : 
输入 了 


SAVE TRANSACTION deletel; 


每 个 保留 点 都 要 取 能 够 标识 它 的 唯一 名 字 ， 以 便 在 回 退 时 ，DBMS 知道 
回 退 到 何 处 。 要 回 退 到 本 例 给 出 的 保留 点 , 在 SQL Server 中 可 如 下 进行 。 





输入 了 
ROLLBACK TRANSACTION deletel; 
在 MariaDB, MySQL 和 Oracle 中 ， 如 下 进行 : 


输入 了 


ROLLBACK TO deletel; 

















下 面 是 一 个 完整 的 SQL Server 例子 : 
输入 


BEGIN TRANSACTION 

INSERT INTO Customers(cust_id, cust_name) 
VALUES('1000000010', 'Toys Emporium'); 

SAVE TRANSACTION StartOrder; 

INSERT INTO Orders(order_num, order_date, cust_id) 
VALUES(20100,'2001/12/1' ,'1000000010'); 
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IF QQERROR <> 0 ROLLBACK TRANSACTION StartOrder; 

INSERT INTO OrderItems(order_num, order_item, prod_id, quantity, 
witem_price) 

VALUES(20100, 1, 'BRO1', 100, 5.49); 

IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; 

INSERT INTO OrderItems(order_num, order_item, prod_id, quantity, 
witem_price) 

VALUES(20100, 2, 'BR03', 100, 10.99); 

IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; 

COMMIT TRANSACTION 


3 Y 


这 里 的 事务 处 理 块 中 包含 了 4 条 INSERT 语句 。 在 第 一 条 INSERT 语句 之 
后 定义 了 一 个 保留 点 ， 因 此 ， 如 果 后 面 的 任何 一 个 INSERT 操作 失败 ， 
事务 处 理 能 够 回 退 到 这 里 。 在 SQL Server 中 ， 可 检查 一 个 名 为 GQERROR 
的 变量 ， 看 操作 是 否 成 功 。( 其 他 DBMS 使 用 不 同 的 函数 或 变量 返回 此 
信息 。) 如 果 @Q@ERROR 返回 一 个 非 0 的 值 , 表示 有 错误 发 生 , 事务 处 理 回 
退 到 保留 点 。 如 果 整 个 事务 处 理 成 功 ， 发 布 COMMIT 以 保留 数据 。 











jn 





























提示 : 保留 点 越 多 越 好 
可 以 在 SQL 代码 中 设置 任意 多 的 保留 点 , 越 多 越 好 。 为 什么 呢 ? 因 为 
保留 点 越 多 ， 你 就 越 能 灵活 地 进行 回 退 。 











20.3 小 结 


这 一 课 介绍 了 事务 是 必须 完整 执行 的 SQL 语句 块 。 我 们 学 习 了 如 何 使 用 
COMMIT 和 ROLLBACK 语句 对 何 时 写 数据 、 何 时 撤销 进行 明确 的 管理 ; 还 
学 习 了 如 何 使 用 保留 点 ， 更 好 地 控制 回 退 操作 。 事 务 处 理 是 个 相当 重要 
的 主题 ， 一 课 内 容 无 法 全 部 涉及 。 各 种 DBMS 对 事务 处 理 的 实现 不 同 ， 
详细 内 容 请 参考 具体 的 DBMS 文档 。 
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这 一 课 将 讲授 什么 是 游标 ， 如 何 使 用 游标 。 


21.1 游标 


SQL 检索 操作 返回 一 组 称 为 结果 集 的 行 , 这 组 返回 的 行 都 是 与 SQL 语句 
相 匹 配 的 行 〈 零 行 或 多 行 )。 简 单 地 使 用 SELECT 语句 ， 没 有 办 法 得 到 第 
一 行 、 下 一 行 或 前 10 行 。 但 这 是 关系 DBMS 功能 的 组 成 部 分 。 



































结果 集 ( result set ) 
SQL 查询 所 检索 出 的 结果 














有 时 ， 需 要 在 检索 出 来 的 行 中 前 进 或 后 退 一 行 或 多 行 ， 这 就 是 游标 的 用 
途 所 在 。 游 标 (cursor) 是 一 个 存储 在 DBMS 服务 器 上 的 数据 库 查 询 ， 
它 不 是 一 条 SELECT 语句 ， 而 是 被 该 语句 检索 出 来 的 结果 集 。 在 存储 了 
游标 之 后 ， 应 用 程序 可 以 根据 需要 滚动 或 浏览 其 中 的 数据 。 















































不 同 的 DBMS 支持 不 同 的 游标 选项 和 特性 。 常 见 的 一 些 选项 和 特性 
如 下 。 


O 能 够 标记 游标 为 只 读 ， 使 数据 能 读 取 ， 但 不 能 更 新 和 删除 。 
O 能 控制 可 以 执行 的 定向 操作 (向 前 、 向 后 、 第 一 、 最 后 、 绝 对 位 置 、 
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相对 位 置 等 )。 
口 能 标记 某 些 列 为 可 编辑 的 ， 茶 些 列 为 不 可 编辑 的 。 
O 规定 范围 ， 使 游标 对 创建 它 的 特定 请 求 〈 如 存储 过 程 ) 或 对 所 有 请 求 
可 访问 。 
O 指示 DBMS 对 检索 出 的 数据 〈 而 不 是 指出 表 中 活动 数据 ) 进行 复制 ， 
使 数据 在 游标 打开 和 访问 期 间 不 变化 。 

































































说 明 : 具体 DBMS 的 支持 
Microsoft Access 不 支持 游标 , 所 以 本 课 的 内 容 不 适用 于 Microsoft Access。 


MySQL 5 已 经 支持 存储 过 程 。 因 此 ， 本 课 的 内 容 不 适用 MySQL 较 早 
的 版 本 。 


的 基本 概念 适用 于 





SQLite 支持 的 游标 称 为 步骤 (step )， 本 课 讲 述 
SQLite 的 步骤， 但 语法 可 能 完全 不 同 。 











游标 主要 用 于 交互 式 应 用 ， 其 中 用 户 需要 滚动 屏幕 上 的 数据 ， 并 对 数据 
进行 浏览 或 做 出 更 改 。 








说 明 : 游标 与 基于 Web 的 应 用 

游标 对 基于 Web 的 应 用 ( 如 ASP, ASP.NET, ColdFusion, PHP, Python, 
Ruby, JSP 等 ) 用 处 不 大 。 虽 然 游 标 在 客户 端 应 用 和 服务 器 会 话 期 间 
存在 , 但 这 种 客户 /服务 器 模式 不 适合 Web 应 用 , 因为 应 用 服务 器 是 数 
据 库 客户 端 而 不 是 最 终 用 户 。 所 以 ， 大 多 数 Web 应 用 开发 人 员 不 使 用 
游标 ， 他 们 根据 自己 的 需要 重新 开发 相应 的 功能 。 











21.2 ”使 用 游标 
使 用 游标 涉及 几 个 明确 的 步 又 。 
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D 在 使 用 游标 前 ,必须 声明 ( 定义 ) 它 。 这 个 过 程 实际 上 没有 检索 数据 ， 
它 只 是 定义 要 使 用 的 SELECT 语句 和 游标 选项 。 

口 一 旦 声明 , 就 必须 打开 游标 以 供 使 用 。 这 个 过 程 用 前 面 定义 的 SELECT 
语句 把 数据 实际 检索 出 来 。 

口 对 于 填 有 数据 的 游标 ， 根 据 需 要 取出 ( 检索 ) 各 行 。 

O 在 结束 游标 使 用 时 ， 必 须 关 闭 游标 ， 可 能 的 话 ， 释 放 游 标 ( 有 赖 于 具 
体 的 DBMS )。 





























声明 游标 后 ， 可 根据 需要 频繁 地 打开 和 关闭 游标 。 在 游标 打开 时 ， 可 根 
据 需 要 频繁 地 执行 取 操 作 。 


21.2.1 创建 游标 


使 用 DECLARE 语句 创建 游标 ， 这 条 语句 在 不 同 的 DBMS 中 有 所 不 同 。 
DECLARE 命名 游标 ， 并 定义 相应 的 SELECT 语句 ， 根 据 需要 带 WHERE 和 
其 他 子 句 。 为 了 说 明 ， 我 们 创建 一 个 游标 来 检索 没有 电子 邮件 地 址 的 所 
有 顾客 ， 作 为 应 用 程序 的 组 成 部 分 ， 帮 助 操作 人 员 找 出 空缺 的 电子 邮件 
地 址 。 















































下 面 是 创建 此 游标 的 DB2、MariaDB MySQL 和 SQL Server 版 本 。 
输入 了 


DECLARE CustCursor CURSOR 
FOR 

SELECT * FROM Customers 
WHERE cust_email IS NULL 














下 面 是 Oracle 和 PostgreSQL 版 本 : 
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输入 了 


DECLARE CURSOR CustCursor 
IS 

SELECT * FROM Customers 
WHERE cust_email IS NULL 


BY 


在 上 面 两 个 版 本 中 ，DECLARE 语句 用 来 定义 和 命名 游标 ， 这 里 为 
CustCursor。SELECT 语句 定义 一 个 包含 没有 电子 邮件 地 址 ( NULL 值 ) 
的 所 有 顾客 的 游标 。 




















定义 游标 之 后 ， 就 可 以 打开 它 了 。 


21.2.2 ”使 用 游标 


使 用 OPEN CURSOR 语句 打开 游标 ， 这 条 语句 很 简单 ， 在 大 多 数 DBMS 
中 的 语法 相同 : 








输入 了 


OPEN CURSOR CustCursor 


uM 
在 处 理 OPEN CURSOR 语句 时 ， 执 行 查询 ， 存 储 检 索 出 的 数据 以 供 浏览 和 
滚动 。 


现在 可 以 用 FETCH 语句 访问 游标 数据 了 。FETCH 指出 要 检索 哪些 行 ， 从 
何 处 检索 它们 以 及 将 它们 放 于 何 处 (如 变量 名 ) 第 一 个 例子 使 用 Oracle 
语法 从 游标 中 检索 一 行 (第 一 行 ): 
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输入 了 


DECLARE TYPE CustCursor IS REF CURSOR 
RETURN Customers%ROWTYPE ; 
DECLARE CustRecord Customers%ROWTYPE 
BEGIN 
OPEN CustCursor ; 
FETCH CustCursor INTO CustRecord ; 
CLOSE CustCursor; 
END; 


BY 


在 这 个 例子 中 ，FETCH 用 来 检索 当前 行 〈 自动 从 第 一 行 开 始 )， 放 到 声明 
的 变量 CustRecord 中 。 对 于 检索 出 来 的 数据 不 做 任何 处 理 。 


下 一 个 例子 (也 使 用 Oracle 语法 ) 中 ， 从 第 一 行 到 最 后 一 行 ， 对 检索 出 
来 的 数据 进行 循环 : 
输入 了 


DECLARE TYPE CustCursor IS REF CURSOR 
RETURN Customers%ROWTYPE ; 
DECLARE CustRecord Customers%ROWTYPE 
BEGIN 
OPEN CustCursor; 
LOOP 
FETCH CustCursor INTO CustRecord; 
EXIT WHEN CustCursor%NOTFOUND; 








END LOOP; 
CLOSE CustCursor; 
END ; 


BY 


与 前 一 个 例子 一 样 ， 这 个 例子 使 用 FETCH 检索 当前 行 ， 放 到 一 个 名 为 
CustRecord 的 变量 中 。 但 不 一 样 的 是 , 这 里 的 FETCH 位 于 LOOP 内 ， 
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此 它 反复 执行 。 代 码 EXIT WHEN CustCursor%NOTFOUND 使 在 取 不 出 更 
多 的 行 时 终止 处 理 (退出 循环 )。 这 个 例子 也 没有 做 实际 的 处 理 ， 实 际 例 
子 中 可 用 具体 的 处 理 代 码 替 换 占 位 符 ...。 


























下 面 是 另 一 个 例子 ， 这 次 使 用 Microsoft SQL Server 语法 : 


输入 了 


DECLARE @cust_id CHAR(10), 
@cust_name CHAR(50), 
@cust_address CHAR(50), 
@cust_city CHAR(50), 
@cust_state CHAR(5), 
@cust_zip CHAR(10), 
@cust_country CHAR(50), 
@cust_contact CHAR(50), 
@cust_email CHAR(255) 
OPEN CustCursor 
FETCH NEXT FROM CustCursor 
INTO @cust_id, @cust_name, @cust_address, 
@cust_city, @cust_state, @cust_zip, 
@cust_country, @cust_contact, @cust_email 
WHILE @@FETCH_STATUS = 0 
BEGIN 


FETCH NEXT FROM CustCursor 
INTO @cust_id, @cust_name, @cust_address, 
@cust_city, @cust_state, @cust_zip, 
@cust_country, @cust_contact, @cust_email 
END 
CLOSE CustCursor 


BY 


在 此 例 中 ， 为 每 个 检索 出 的 列 声 明 一 个 变量 ，FETCH 语句 检索 一 行 并 保 
存 值 到 这 些 变量 中 。 使 用 WHILE 循环 处 理 每 一 行 ， 条 件 WHILE 
@@FETCH_STATUS = 0 在 取 不 出 更 多 的 行 时 终止 处 理 (退出 循环 )。 这 个 
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例子 也 不 进行 具体 的 处 理 ， 实 际 代 码 中 ， 应 该 用 具体 的 处 理 代码 替换 其 
中 的 .. . 占 位 符 。 





21.2.3 关闭 游标 


如 前 面 几 个 例子 所 述 ， 游 标 在 使 用 完毕 时 需要 关闭 。 此 外 ，SQL Server 
等 DBMS 要 求 明 确 释 放 游 标 所 占用 的 资源 。 下 面 是 DB2、Oracle 和 
PostgreSQL 的 语法 。 























输入 了 


CLOSE CustCursor 

















下 面 是 Microsoft SQL Server 的 版 本 。 


输入 了 


CLOSE CustCursor 
DEALLOCATE CURSOR CustCursor 


BY 


CLOSE 语句 用 来 关闭 游标 。 一 旦 游标 关闭 ， 如 果 不 再 次 打开 ， 将 不 能 
用 。 第 二 次 使 用 它 时 不 需要 再 声明 ， 只 需 用 OPEN 打开 它 即 可 。 





21.3 ”小结 

我 们 在 本 课 讲 授 了 什么 是 游标 ， 为 什么 使 用 游标 。 你 使 用 的 DBMS 可 能 
会 提供 某 种 形式 的 游标 ， 以 及 这 里 没有 提 及 的 功能 。 更 详细 的 内 容 请 参 
阅 具 体 的 DBMS 文档 。 
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第 22 课 高 级 SQL 特性 


这 一 课 介 绍 SQL 所 涉及 的 几 个 高 级 数据 处 理 特性 : 约束 、 索 引 和 触发 顺 。 


22.1 ZJ 


* 





SQL 已 经 改进 过 多 个 版 本 ， 成 为 非常 完善 和 强大 的 语言 。 许 多 强 有 力 的 
特性 给 用 户 提供 了 高 级 的 数据 处 理 技术 ， 如 约束 。 





关联 表 和 引用 完整 性 已 经 在 前 面 讨 论 过 几 次 。 正 如 所 述 ， 关 系数 据 库存 储 
分 解 为 多 个 表 的 数据 ， 每 个 表 存储 相应 的 数据 。 利 用 键 来 建立 从 一 个 表 到 


男 一 个 表 的 3 


正确 地 进行 关系 数据 库 设计 , 需要 一 种 方法 保证 只 在 表 中 撒 








| 用 (由 此 产生 了 术语 引用 完整 性 (referential integrity ) )。 








入 合法 数据 。 


例如 , 如 果 Orders 表 存 储 订 单 信息 , OrderItems 表 存 储 订 单 详细 内 容 ， 
应 该 保证 OrderItems 中 引用 的 任何 订单 ID 都 存在 于 Orders 中 。 类 似 
地 ,在 Orders 表 中 引用 的 任意 顾客 必须 存在 于 Customers 表 中 。 


虽然 可 以 在 捐 











有 入 新 行 时 进行 检查 ( 在 男 一 个 表 上 执行 SELECT, ATAU 


EHT 











有 值 合法 并 存在 )， 但 最 好 不 要 这 样 做 ， 原 因 如 下 。 











口 如 果 在 客户 端 层面 上 实施 数据 库 完 整 性 规则 ， 则 每 个 客户 端 都 要 被 迫 
实施 这 些 规则 ， 一 定 会 有 一 些 客户 端 不 实施 这 些 规则 。 
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D 执行 客户 端 检查 是 非常 耗 时 的 ， 而 DBMS 执行 这 些 检查 会 相对 高 效 。 











约束 ( constraint ) 
管理 如 何 播 入 或 处 理 数 据 库 数据 的 规则 。 











DBMS 通过 在 数据 库 表 上 施加 约束 来 实施 引用 完整 性 。 大 多 数 约束 是 在 
表 定 义 中 定义 的 ， 如 第 17 课 所 述 ， 用 CREATE TABLE 或 ALTER TABLE 
语句 。 














注意 : 具体 DBMS 的 约束 

有 几 种 不 同类 型 的 约束 ， 每 个 DBMS 都 提供 自己 的 支持 。 因 此 ， 这 里 
给 出 的 例子 在 不 同 的 DBMS 上 可 能 有 不 同 的 反应 。 在 进行 试验 之 前 ， 
请 参阅 具体 的 DBMS 文档 。 











22.1.1 主键 


我 们 在 第 1 课 简 单 提 过 主键 ,。 主键 是 一 种 特殊 的 约束 , 用 来 保证 一 列 ( 或 
一 组 列 ) 中 的 值 是 唯一 的 ， 而 且 永 不 改动 。 换 句 话说 ， 表 中 的 一 列 (或 
多 个 列 ) 的 值 唯一 标识 表 中 的 每 一 行 。 这 方便 了 直接 或 交互 地 处 理 表 中 
的 行 。 没有 主键 , 要 安全 地 UPDATE 或 DELETE 特定 行 而 不 影响 其 他 行 会 
非常 困难 。 



























































表 中 任意 列 只 要 满足 以 下 条 件 ， 都 可 以 用 于 主键 。 





口 任意 两 行 的 主键 值 都 不 相同 。 

口 每 行 都 具有 一 个 主键 值 ( 即 列 中 不 允许 NULL 值 )。 

口 包含 主键 值 的 列 从 不 修改 或 更 新 。( 大 多 数 DBMS 不 允许 这 么 做 ,但 
如 果 你 使 用 的 DBMS 允许 这 样 做 ， 好 吧 ， 千 万 别 ! ) 
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生 


























口 主键 值 不 能 重用 。 如 果 从 表 中 删除 某 一 行 ， 其 主键 值 不 分 配给 新 行 。 




















一 种 定义 主键 的 方法 是 创建 它 ， 如 下 所 示 。 


输入 

CREATE TABLE Vendors 

( 
vend_id CHAR(10) NOT NULL PRIMARY KEY, 
vend_name CHAR(50) NOT NULL, 
vend_address CHAR(50) NULL, 
vend_city CHAR(50) NULL, 
vend_state CHAR(5) NULL, 
vend_zip CHAR(10) NULL, 
vend_country CHAR(50) NULL 

) ; 

S Y 





在 此 例子 中 ， 给 表 的 vend_id 列 定 义 添 加 关键 字 PRIMARY KEY， 使 其 成 





输入 了 


ALTER TABLE Vendors 
ADD CONSTRAINT PRIMARY KEY (vend_id); 


3 Y 


这 里 定义 相同 的 列 为 主键 ， 但 使 用 的 是 CONSTRAINT 语法 。 此 语法 也 可 
以 用 于 CREATE TABLE 和 ALTER TABLE 语句 。 








说 明 : SQLite 中 的 键 
SQLite 不 允许 使 用 ALTER TABLE 定义 键 ,要 求 在 初始 的 CREATE TABLE 
语句 中 定义 它们 。 
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22.1.2 外 键 








外 键 是 表 中 的 一 列 ， 其 值 必须 列 在 男 一 表 的 主键 中 。 外 键 是 保证 引用 完 
整 性 的 极其 重要 部 分 。 我 们 举 个 例子 来 理解 外 键 。 





Orders 表 将 录入 到 系统 的 每 个 订单 作为 一 行 包含 其 中 。 顾客 信息 存储 在 
Customers XF, Orders 表 中 的 订单 通过 顾客 ID 与 Customers 表 中 的 
特定 行 相 关联 。 顾 客 ID 为 Customers 表 的 主键 ， 每 个 顾客 都 有 唯一 的 
ID。 订 单 号 为 0rders 表 的 主键 ， 每 个 订单 都 有 唯一 的 订单 号 。 









































Orders 表 中 顾客 ID 列 的 值 不 一 定 是 唯一 的 。 如 果 某 个 顾客 有 多 个 订单 ， 
则 有 多 个 行 具 有 相同 的 顾客 ID (虽然 每 个 订单 都 有 不 同 的 订单 号 )。 同 
BF, Orders 表 中 顾客 ID 列 的 合法 值 为 Customers 表 中 顾客 的 ID。 





























这 就 是 外 键 的 作用 。 在 这 个 例子 中 , 在 Orders 的 顾客 ID 列 上 定义 了 一 
个 外 键 ， 因 此 该 列 只 能 接受 Customers 表 的 主键 值 。 




















下 面 是 定义 这 个 外 键 的 方法 。 

输入 

CREATE TABLE Orders 

( 
order_num INTEGER NOT NULL PRIMARY KEY, 
order_date DATETIME NOT NULL, 
cust_id CHAR(10) NOT NULL REFERENCES 
wCustomers(cust_id) 

); 

SH Y 





其 中 的 表 定 义 使 用 了 REFERENCES 关键 字 ， 它 表示 cust_id 中 的 任何 值 
都 必须 是 Customers 表 的 cust_id 中 的 值 。 
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相同 的 工作 也 可 以 在 ALTER TABLE 语句 中 用 CONSTRAINT 语法 来 完成 : 





输入 了 


ALTER TABLE Orders 
ADD CONSTRAINT 
FOREIGN KEY (cust_id) REFERENCES Customers (cust_id) 





提示 : 外 键 有 助 防止 意外 删除 

如 第 6 课 所 述 ， 除 帮助 保证 引用 完整 性 外 ， 外 键 还 有 另 一 个 重要 作用 。 
人 DBMS 不 允许 删除 在 另 一 个 表 中 具有 关联 行 的 行 。 例 
如 ， 不 能 删除 关联 订单 的 顾客 。 删 除 该 顾客 的 唯一 方法 是 首先 删除 相 
i (这 表示 还 要 删除 相关 的 订单 项 )。 由 于 需要 一 系列 的 删除 ， 
因而 利用 外 键 可 以 防止 意外 删除 数据 。 


有 的 DBMS 支持 称 为 级 联 删 除 (cascading delete ) 的 特性 。 如 果 启 用 ， 
该 特性 在 从 一 个 表 中 删除 行 时 删除 所 有 相关 的 数据 。 例 如 ， 如 果 启 用 
级 联 删 除 并 且 从 Customers 表 中 删除 某 个 顾客 , 则 任何 关联 的 订单 行 
也 会 被 自动 删除 。 








22.1.3 ”唯一 约束 


一 约束 用 来 保证 一 列 〈 或 一 组 列 ) 中 的 数据 是 唯一 的 。 它 们 类 似 于 主 
键 ， 但 存在 以 下 重要 区 别 。 


























口 表 可 包含 多 个 唯一 约束 ， 但 每 个 表 只 允许 一 个 主键 。 
口 唯一 约束 列 可 包含 NULL 值 。 

口 唯一 约束 列 可 修改 或 更 新 。 

口 唯一 约束 列 的 值 可 重复 使 用 。 

D 与 主键 不 一 样 ， 唯 一 约束 不 能 用 来 定义 外 键 。 
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employees 表 是 一 个 使 用 约束 的 例子 。 每 个 雇员 都 有 唯一 的 社会 安全 号 ， 
但 我 们 并 不 想 用 它 作 主键 ， 因 为 它 太 长 〈 而 且 我 们 也 不 想 使 该 信息 容易 
利用 )。 因此 , 每 个 雇员 除了 其 社会 安全 号 外 还 有 唯一 的 雇员 ID (主键 )。 



































雇员 ID 是 主键 ， 可 以 确定 它 是 唯一 的 。 你 可 能 还 想 使 DBMS 保证 每 个 
社会 安全 号 也 是 唯一 的 ( 保证 输入 错误 不 会 导致 使 用 他 人 号 码 )。 可 以 通 
过 在 社会 安全 号 列 上 定义 UNIQUE 约束 做 到 。 

















唯一 约束 的 语法 类 似 于 其 他 约束 的 语法 。 唯 一 约束 既 可 以 用 UNIQUE 关 
键 字 在 表 定 义 中 定义 ， 也 可 以 用 单独 的 CONSTRAINT 定义 。 








22.1.4 检查 约束 


检查 约束 用 来 保证 一 列 ( 或 一 组 列 ) 中 的 数据 满足 一 组 指定 的 条 件 。 检 
查 约 束 的 常见 用 途 有 以 下 几 点 。 





口 检查 最 小 或 最 大 值 。 例如 , 防止 0 个 物品 的 订单 (即使 0 是 合法 的 数 )。 
口 指定 范围 。 例 如 ， 保 证 发 货 日 期 大 于 等 于 今天 的 日 期 ， 但 不 超过 今天 
起 一 年 后 的 日 期 。 

O 只 允许 特定 的 值 。 例 如 ， 在 性 别 字段 中 只 允许 M sk F. 

















换 句 话说 , 第 1 课 介 绍 的 数据 类 型 限制 了 列 中 可 保存 的 数据 的 类 型 。 检 查 
约束 在 数据 类 型 内 又 做 了 进一步 的 限制 , 这 些 限 制 极其 重要 , 可 以 确保 插 
入 数据 库 的 数据 正 是 你 想 要 的 数据 ,不 需要 依赖 于 客户 端 应 用 程序 或 用 户 
来 保证 正确 获取 它 ，DBMS 本 身 将 会 拒绝 任何 无 效 的 数据 。 


























下 面 的 例子 对 0rderItems 表 施 加 了 检查 约束 ， 它 保证 所 有 物品 的 数量 
大 于 0。 
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输入 了 

CREATE TABLE OrderItems 

( 
order_num INTEGER NOT NULL, 
order_item INTEGER NOT NULL, 
prod_id CHAR(10) NOT NULL, 
quantity INTEGER NOT NULL CHECK (quantity > 0), 
item_price MONEY NOT NULL 

); 

5 Y 








利用 这 个 约束 ， 任 何 插入 (或 更 新 ) 的 行 都 会 被 检查 ， 保 证 quantity 
大 于 0。 

















检查 名 为 gender 的 列 只 包含 M 或 F, 可 编写 如 下 的 ALTER TABLE 语句 : 


输入 了 


ADD CONSTRAINT CHECK (gender LIKE '[MF]') 





提示 : 用 户 定 义 数 据 类 型 

有 的 DBMS 允许 用 户 定义 自己 的 数据 类 型 。 它们 是 定义 检查 约束 (或 
其 他 约束 ) 的 基本 简单 数据 类 型 ,例如 ,你 可 以 定义 自己 的 名 为 gender 
的 数据 类 型 ， 它 是 单字 符 的 文本 数据 类 型 ， 带 限制 其 值 为 M 或 F( 对 
于 未 知 值 或 许 还 允许 NULL ) 的 检查 约束 。 然 后 ， 可 以 将 此 数据 类 型 用 
于 表 的 定义 。 定 制 数据 类 型 的 优点 是 只 需 施 加 约束 一 次 (在 数据 类 型 
定义 中 )， 而 每 当 使 用 该 数据 类 型 时 ， 都 会 自动 应 用 这 些 约 束 。 请 查阅 
相应 的 DBMS 文档 ， 看 它 是 否 支 持 自 定义 数据 类 型 。 
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22.2 索引 


索引 用 来 排序 数据 以 加 快 搜索 和 排序 操作 的 速度 。 想 像 一 本 书后 的 索引 
《如 本 书后 的 索引 )， 可 以 帮助 你 理解 数据 库 的 索引 。 























假如 要 找 出 本 书 中 所 有 的 “数据 类 型 ”这 个 词 ， 简 单 的 办 法 是 从 第 1 页 
开始 ， 浏 览 每 一 行 。 虽然 这 样 做 可 以 完成 任务 , 但 显然 不 是 一 种 好 的 办 
法 。 浏览 少数 几 页 文字 可 能 还 行 , 但 以 这 种 方式 浏览 整 部 书 就 不 可 行 了 。 
随 着 要 搜索 的 页 数 不 断 增加 ， 找 出 所 需 词汇 的 时 间 也 会 增加 。 











这 就 是 书籍 要 有 索引 的 原因 。 索 引 按 字母 顺序 列 出 词汇 及 其 在 书 中 的 位 
置 。 为 了 搜索 “数据 类 型 ”一 词 ， 可 在 索引 中 找 出 该 词 ， 确 定 它 出 现在 
哪些 页 中 。 然 后 再 翻 到 这 些 页 ， 找 出 “数据 类 型 ”一 词 。 









































使 索引 有 用 的 因素 是 什么 ?” 很 简单 ， 就 是 恰当 的 排序 。 找 出 书 中 词汇 的 
困难 不 在 于 必须 进行 多 少 搜索 ， 而 在 于 书 的 内 容 没 有 按 词汇 排序 。 如 果 
书 的 内 容 像 字典 一 样 排序 ， 则 索引 没有 必要 ( 因此 字典 就 没有 索引 )。 




















数据 库 索 引 的 作用 也 一 样 。 主 键 数据 总 是 排序 的 ， 这 是 DBMS 的 工作 。 
因此 ， 按 主键 检索 特定 行 总 是 一 种 快速 有 效 的 操作 。 








但 是 ， 搜 索 其 他 列 中 的 值 通常 效率 不 高 。 例 如 ， 如 果 想 搜索 住 在 某 个 州 的 
客户 , 怎么 办 ? 因为 表 数 据 并 未 按 州 排序 , DBMS 必须 读 出 表 中 所 有 行 (从 
第 一 行 开 始 )， 看 其 是 否 匹 配 。 这 就 像 要 从 没有 索引 的 书 中 找 出 词汇 一 样 。 











解决 方法 是 使 用 索引 。 可 以 在 一 个 或 多 个 列 上 定义 索引 ,使 DBMS 保存 
其 内 容 的 一 个 排 过 序 的 列表 。 在 定义 了 索引 后 ，DBMS 以 使 用 书 的 索引 
类 似 的 方法 使 用 它 。DBMS 搜索 排 过 序 的 索引 ， 找 出 匹配 的 位 置 ， 然 后 


ORRAT, 
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在 开始 创建 索引 前 ， 应 该 记 住 以 下 内 容 。 


O 索引 改善 检索 操作 的 性 能 ， 但 降低 了 数据 插入 、 修 改 和 删除 的 性 能 。 

在 执行 这 些 操作 时 ，DBMS 必须 动态 地 更 新 索引 。 

口 索引 数据 可 能 要 占用 大 量 的 存储 空间 。 

口 并 非 所 有 数据 都 适合 做 索引 。 取 值 不 多 的 数据 〈 如 州 ) 不 如 具有 更 多 
可 能 值 的 数据 ( 如 姓 或 名 )， 能 通过 索引 得 到 那么 多 的 好 处 。 

口 索引 用 于 数据 过 滤 和 数据 排序 。 如 果 你 经 常 以 某 种 特定 的 顺序 排序 数 
据 ， 则 该 数据 可 能 适合 做 索引 。 

口 可 以 在 索引 中 定义 多 个 列 (例如 ， 州 加 上 城市 )。 这 样 的 索引 仅 在 以 州 
加 城市 的 顺序 排序 时 有 用 。 如 果 想 按 城 市 排序 ， 则 这 种 索引 没有 用 处 。 



































没有 严格 的 规则 要 求 什么 应 该 索引 ， 何 时 索引 。 大 多 数 DBMS 提供 了 可 
用 来 确定 索引 效率 的 实用 程序 ， 应 该 经 常 使 用 这 些 实用 程序 。 




















索引 用 CREATE INDEX 语 名 创建 (不 同 DBMS 创建 索引 的 语句 变化 很 大 )。 
下 面 的 语句 在 Products 表 的 产品 名 列 上 创建 一 个 简单 的 索引 。 

输入 了 

CREATE INDEX prod_name_ind 

ON Products (prod_name) ; 

S Y 


索引 必须 唯一 命名 。 这 里 的 索引 名 prod_name_ind 在 关键 字 CREATE 
INDEX 之 后 定义 。ON 用 来 指定 被 索引 的 表 ， 而 索引 中 包含 的 列 〈 此 例 中 
仅 有 一 列 ) 在 表 名 后 的 圆 括号 中 给 出 。 
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提示 : 检查 索引 

索引 的 效率 随 表 数据 的 增加 或 改变 而 变化 。 许 多 数据 库 管 理 员 发 现 ， 
过 去 创建 的 某 个 理想 的 索引 经 过 几 个 月 的 数据 处 理 后 可 能 变 得 不 再 理 
想 了 。 最 好 定期 检查 索引 ， 并 根据 需要 对 索引 进行 调整 。 











22.3 触发 器 


触发 器 是 特殊 的 存储 过 程 ， 它 在 特定 的 数据 库 活 动 发 生 时 自动 执行 。 触 发 
器 可 以 与 特定 表 上 的 INSERT、UPDATE 和 DELETE 操作 (或 组 合 ) 相关 联 。 


























与 存储 过 程 不 一 样 〈 存 储 过 程 只 是 简单 的 存储 SQL 语句 )， 触 发 器 与 单 
个 的 表 相 关联 。 与 Orders 表 上 的 INSERT 操作 相关 联 的 触发 器 只 在 
Orders 表 中 插入 行 时 执行 。 类 似 地 ，Customers 表 上 的 INSERT 和 
UPDATE 操作 的 触发 器 只 在 表 上 出 现 这 些 操作 时 执行 。 








触发 器 内 的 代码 具有 以 下 数据 的 访问 权 : 


口 INSERT 操作 中 的 所 有 新 数据 ; 
口 UPDATE 操作 中 的 所 有 新 数据 和 旧 数 据 ; 
口 DELETE 操作 中 删除 的 数据 。 








根据 所 使 用 的 DBMS 的 不 同 ,触发 需 可 在 特定 操作 执行 之 前 或 之 后 执行 。 




















下 面 是 触发 需 的 一 些 常见 用 途 。 





口 保证 数据 一 致 。 例 如 ,在 INSERT 或 UPDATE 操作 中 将 所 有 州 名 转换 
为 大 写 。 

D 基于 某 个 表 的 变动 在 其 他 表 上 执行 活动 。 例 如 ， 每 当 更 新 或 删除 一 行 
时 将 审计 跟踪 记录 写 和 人 某 个 日 志 表 。 
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O 进行 额外 的 验证 并 根据 需要 回 退 数据 。 例 如 ， 保 证 某 个 顾客 的 可 用 资 
金 不 超 限 定 ， 如 果 已 经 超出 ， 则 阻塞 插入 。 
口 计算 计算 列 的 值 或 更 新 时 间 检 。 
































读者 可 能 已 经 注意 到 了 ,不 同 DBMS 的 触发 器 创建 语法 差异 很 大 ,更 详 
细 的 信息 请 参阅 相应 的 文档 。 





下 面 的 例子 创建 一 个 触发 器 ， 它 对 所 有 INSERT 和 UPDATE 操作 ,将 
Customers 表 中 的 cust_state 列 转换 为 大 写 。 


这 是 本 例子 的 SQL Server 版 本 。 
输入 了 


CREATE TRIGGER Customer_state 

ON Customers 

FOR INSERT, UPDATE 

AS 

UPDATE Customers 

SET cust_state = Upper(cust_state) 

WHERE Customers.cust_id = inserted.cust_id; 


这 是 本 例子 的 Oracle 和 PostgreSQL 的 版 本 : 
输入 了 


CREATE TRIGGER Customer_state 

AFTER INSERT OR UPDATE 

FOR EACH ROW 

BEGIN 

UPDATE Customers 

SET cust_state = Upper(cust_state) 
WHERE Customers.cust_id = :OLD.cust_id 
END; 
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提示 : 约束 比 触发 器 更 快 
一 般 来 说 ， 约 束 的 处 理 比 触发 器 快 ， 因 此 在 可 能 的 时 候 ， 应 该 尽量 使 
用 约束 。 











224 数据 库 安 全 


对 于 组 织 来 说 , 没有 什么 比 它 的 数据 更 重要 了 , 因此 应 该 保护 这 些 数据 ， 
使 其 不 被 偷盗 或 任意 浏览 。 当 然 ， 数 据 也 必须 允许 需要 访问 它 的 用 户 访 
问 ， 因 此 大 多 数 DBMS 都 给 管理 员 提 供 了 管理 机 制 ， 利 用 管理 机 制 授 予 
或 限制 对 数据 的 访问 。 




















任何 安全 系统 的 基础 都 是 用 户 授权 和 身份 确认 。 这 是 一 种 处 理 ， 通 过 这 
种 处 理 对 用 户 进行 确认 , 保证 他 是 有 权 用 户 , 允许 执行 他 要 执行 的 操作 。 
有 的 DBMS 为 此 结合 使 用 了 操作 系统 的 安全 措施 ,而 有 的 维护 自己 的 用 
户 及 密码 列表 ， 还 有 一 些 结合 使 用 外 部 目录 服务 服务 器 。 








一 般 说 来 ， 需 要 保护 的 操作 有 : 


O 对 数据 库 管理 功能 〈 创建 表 、 更 改 或 删除 已 存在 的 表 等 ) 的 访问 ; 
口 对 特定 数据 库 或 表 的 访问 ; 

O 访问 的 类 型 (只 读 、 对 特定 列 的 访问 等 ); 

口 仅 通过 视图 或 存储 过 程 对 表 进 行 访问 ; 

口 创建 多 层次 的 安全 措施 ， 从 而 允许 多 种 基于 登录 的 访问 和 控制 
口 限制 管理 用 户 账 号 的 能 






































安全 性 使 用 SQL 的 GRANT 和 REVOKE 语句 来 管理 , 不 过 , 大 多 数 DBMS 
提供 了 交互 式 的 管理 实用 程序 ， 这 些 实用 程序 在 内 部 使 用 GRANT 和 
REVOKE 语句 。 
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本 课 讲授 如 何 使 用 SQL 的 一 些 高 级 特性 。 约束 是 实施 引用 完整 性 的 重要 
部 分 ， 索 引 可 改善 数据 检索 的 性 能 ， 触 发 需 可 以 用 来 执行 运行 前 后 的 处 
理 ， 安 全 选项 可 用 来 管理 数据 访问 。 不 同 的 DBMS 可 能 会 以 不 同 的 形式 
提供 这 些 特性 ， 更 详细 的 信息 请 参阅 具体 的 DBMS 文档 。 
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编写 SQL 语句 需要 良好 地 理解 基本 数据 库 设 计 。 如 果 不 知 道 什么 信息 存 
放 在 什么 表 中 ， 表 与 表 之 间 如 何 互 相关 联 ， 行 中 数据 如 何 分 解 ， 那 么 要 
编写 高 效 的 SQL 是 不 可 能 的 。 








强烈 建议 读者 实际 练习 本 书 的 每 个 例子 。 所 有 课 都 共同 使 用 了 一 组 数据 
文件 。 为 帮助 你 更 好 地 理解 这 些 例子 、 学 好 各 课 内 容 ， 本 附录 描述 了 所 
用 的 表 、 表 之 间 的 关系 以 及 如 何 创建 (或 获得 ) 它们 。 





A.1 样 例 表 


本 书 中 所 用 的 表 是 一 个 假想 玩具 经 销 商 使 用 的 订单 录入 系统 的 组 成 部 
分 。 这 些 表 用 来 完成 以 下 几 项 任务 : 



































D 管理 供应 商 ; 

口 管理 产品 目录 ; 
口 管理 顾客 列表 ; 
口 录入 顾客 订单 。 











完成 它们 需要 5 个 表 ( 它们 作为 一 个 关系 数据 库 设 计 的 组 成 部 分 紧密 关 
联 )。 以 下 各 节 给 出 每 个 表 的 描述 。 
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说 明 : 简化 的 例子 

这 里 使 用 的 表 不 完整 ， Oe 5 O 
有 的 大 量 数据 ( 如 工资 和 记 账 信息 、 发 货 追 踪 信 息 等 ), 不过， 这 些 表 
a 
将 这 些 技术 用 于 自己 的 数据 库 。 











表 的 描述 
下 面 介绍 5 个 表 及 每 个 表 内 的 列 名 。 


1. Vendors 表 








Vendors 表 存 储 销售 产品 的 供应 商 。 每 个 供应 商 在 这 个 表 中 有 一 个 记录 ， 
供应 商 ID 列 (vend_id ) 用 于 进行 产品 与 供应 商 的 匹配 。 





表 A-1 Vendors 表 的 列 






















































































列 说 明 
vend_id 佳 一 的 供应 商 ID 
vend_name 供应 商 名 
vend_address 供应 商 的 地 志 
vend_city 上 共 应 商 所 在 城市 
vend_state 上 共 应 商 所 在 州 
vend_zip 上 共 应 商 地 址 邮政 编码 
vend_country 上 共 应 商 所 在 国家 





























口 所 有 表 都 应 该 有 主键 。 这 个 表 应 该 用 vend_id 作为 其 主键 。 





2. Products 表 





Products 表 包 含 产品 目录 ， 每 行 一 个 产品 。 每 个 产品 有 唯一 的 ID 
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(prod_id l), 并 且 借 助 vend_id( 供应 商 的 唯一 ID ) 与 供应 商 相 关联 。 


表 A-2 Products 表 的 列 

















列 说 BB 
prod_id 唯一 的 产品 JP 
vend_id 产品 供应 商 ID (关联 到 Vendors 表 的 vend_id ) 
prod_name 产品 名 
prod_price 产品 价格 
prod_desc 产品 描述 





的 vend_id Z|; 


3. Customers 表 





口 所 有 表 都 应 该 有 主键 。 这 个 表 应 该 用 prod_id 作为 其 主键 。 
口 为 实施 引用 完整 性 ,应 该 在 vend_id 上 定义 一 个 外 键 ,关联 到 Vendors 











Customers 表 存 储 所 有 顾客 信息 。 每 个 顾客 有 唯一 的 ID ( cust_id 列 )。 




















表 A-3 Customers 表 的 列 
列 说 BA 

cust_id 唯一 的 顾客 ID 
cust_name 顾客 名 
cust_address 顾客 的 地 址 
cust_city 顾客 所 在 城市 
cust_state 顾客 所 在 州 
cust_zip 顾客 地 址 邮政 编码 
cust_country 顾客 所 在 国家 
cust_contact 顾客 的 联系 名 
cust_email 顾客 的 电子 邮件 地 址 
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口 所 有 表 都 应 该 有 主键 。 这 个 表 应 该 用 cust_id 作为 它 的 主键 。 


4. Orders 表 





Orders 表 存 储 顾客 订单 (不 是 订单 细节 )。 每 个 订单 唯一 编号 
(order_num IJ )。Orders 表 按 cust_id ïi) (关联 到 Customers 表 的 顾 
客 唯 一 ID ) 关联 到 相应 的 顾客 。 


表 A-4 Orders 表 的 列 














列 说 BB 
order_num 唯一 的 订单 号 
order_date 订单 日 期 
cust_id 订单 顾客 ID ( 关联 到 Customers 表 的 cust_id ) 





口 所 有 表 都 应 该 有 主键 。 这 个 表 应 该 用 order_num 作为 其 主键 。 
口 为 实施 引用 完整 性 ， 应 该 在 cust_id 上 定义 一 个 外 键 ， 关 联 到 

















Customers 的 cust_id 列 。 


5. OrderItems £ 


OrderItems 表 存 储 每 个 订单 中 的 实际 物品 ， 每 个 订单 的 每 个 物品 一 行 。 
对 于 Orders 表 的 每 一 行 ， 在 0rderItems 表 中 有 一 行 或 多 行 。 每 个 订 
单 物品 由 订单 号 加 订单 物品 (第 一 个 物品 、 第 二 个 物品 等 ) 唯一 标识 。 

订单 物品 用 order_num 列 (关联 到 Orders 表 中 订单 的 唯一 ID ) 与 其 相 
应 的 订单 相关 联 。 此 外 ， 每 个 订单 物品 包含 该 物品 的 产品 ID (把 物品 关 





联 到 Products 表 )。 
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表 A-5 ”OrderItems 表 的 列 








列 说 明 
order_num 订单 号 (关联 到 Orders 表 的 order_num ) 
order_item 订单 物品 号 (订单 内 的 顺序 ) 
prod_id 产品 ID (关联 到 Products 表 的 prod_id ) 
quantity 物品 数量 
item_price 物品 价格 





口 所 有 表 都 应 该 有 主键 。 这 个 表 应 该 用 order_num 和 order_item 作为 
其 主键 。 
口 为 实施 引用 完整 性 ， 应 该 在 order_num 和 prod_id 上 定义 外 键 ， 关 
联 order_num 到 Orders 的 order_num 列 ,关联 prod_id 到 Products 
的 prod_id 列 。 











数据 库 管 理 员 通 常 使 用 关系 图 来 说 明 数 据 库 表 的 关联 方式 。 要 记 住 ， 正 
如 上 面 表 描述 提 到 的 ， 外 键 定义 了 这 些 关 系 。 图 A-1 是 本 附录 描述 的 五 
个 表 的 关系 图 。 











Customers Orders 














Orderkems 
custid : 9 order_num |  order_num 
cust_name e order_date Ü order_item 
cust_addrass = cust id prod_id 
cust_city quantity 
cust_state item_price 
cust_zip 
cust_country 
cust_contact 
cust_email 
Venclors Products 
[ ç vend_id i aa ¥ prod id 
vend_name vend_id 
vend_address pe nasi 
vend_city prod_price 
vend_state prod desc 
vend zip 
vend_country 


图 A-1 样 例 表 关系 





=l 
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A.2 获得 样 例 表 


学 习 各 个 例子 ， 需 要 一 组 填充 了 数据 的 表 。 所 需要 获得 和 运行 的 东西 都 
可 以 在 本 书 网 页 http://www.forta.com/books/0672336073/ 找 到 。 





A.2.1 下 载 可 供 使 用 的 数据 文件 
可 从 上 述 URL 下 载 一 个 填充 了 数据 的 如 下 格式 的 文件 : 

















口 Apache Open Office Base 
口 Microsof Access ( 2000 和 2007 ) 
OQ SQLite 














如 果 使 用 这 些 文件 ， 不 需要 执行 任何 SQL 创建 和 填充 脚本 。 


A.2.2 下 载 DBMS SQL 脚本 


大 多 数 DBMS 以 不 自己 完成 文件 分 布 的 格式 存储 数据 ( 如 Access、Open 
Office Base 和 SQLite 那样 ). 对 于 这 些 DBMS ,可 以 从 上 述 URL 下 载 SQL 
脚本 。 对 于 每 个 DBMS ， 有 两 个 文件 : 


D create.text 包含 创建 5 个 数据 库 表 ( 包括 定义 所 有 主键 和 外 键 约束 ) 的 
SQL 语句 。 
口 populate.txt 包含 用 来 填充 这 些 表 的 SQL INSERT 语句 。 

















这 些 文件 中 的 SQL 语句 依赖 于 具体 的 DBMS, 因此 应 该 执行 适合 于 你 的 
DBMS 的 那个 。 这 些 脚本 为 方便 读者 而 提供 ， 作 者 对 执行 它们 万 一 引起 
的 问题 不 承担 任何 责任 。 
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在 本 书 付 印 时 ， 有 以 下 脚本 可 供 使 用 : 





B IBM DB2; 

口 Microsoft SQL Server ( 包括 Microsoft SQL Server Express ); 
口 MariaDB 

B MySQL; 

口 Oracle( 包括 Oracle Express ); 

ü PostgreSQL, 








适用 于 其 他 DBMS 的 脚本 可 能 会 根据 需要 或 请 求 而 增加 。 


附录 B 提供 了 在 几 个 流行 环境 中 执行 脚本 的 说 明 。 





说 明 : 创建 ， 然 后 填充 
必须 在 执行 表 填 充 脚本 前 执行 表 创 建 脚 本 。 应 该 检查 这 些 脚 本 返回 的 
错误 。 如 果 创 建 脚本 失败 ， 则 应 该 在 继续 表 填 充 前 解决 存在 的 问题 。 








说 明 : 具体 DBMS 的 设置 指令 
用 于 设置 DBMS 的 具体 步骤 依 使 用 的 DBMS 有 很 大 不 同 。 从 本 书 网 页 
下 载 脚本 或 数据 库 时 ， 你 会 看 到 README 文件 ， 它 提供 了 针对 特定 
DBMS 的 具体 设置 和 安装 步骤 。 
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正如 第 1 课 所 述 ，SQL 不 是 一 个 应 用 ， 而 是 一 种 语言 。 要 学 习 本 书 中 的 
例子 ， 你 需要 一 个 支持 SQL 语句 执行 的 应 用 程序 。 











本 附录 描述 了 在 几 个 最 常用 的 应 用 中 执行 SQL 语句 的 步骤 。 


你 可 以 使 用 下 面 列 出 的 任何 一 个 应 用 或 其 他 应 用 来 测试 和 试验 SQL 代 
码 。 那 么 ， 究 竟 应 该 使 用 哪个 呢 ? 











O 许多 DBMS 具有 自己 的 客户 端 实用 程序 , 它们 通常 是 很 好 的 学 习 出 发 
点 。 但 有 些 没有 直观 的 用 户 界面 。 

口 如 果 你 是 一 位 Web 开发 人 员 ， 那 么 可 以 使 用 任何 服务 器 端 Web 编程 
语言 ， 包括 ASP.NET, ColdFusion, Java, JSP, PHP, Python 和 Ruby 


























on Rails- 
a 有 很 多 第 三 方 工具 和 实用 程序 ， 本 书 网 页 http://www.forta.com/books/ 
0672336073/ 上 有 其 中 的 一 些 的 链接 。 


B.1 使 用 Apache Open Office Base 


Apache Open Office Base 是 一 个 基于 Java 的 开源 客户 端 数 据 库 应 用 ,Open 
Office Base 查询 可 以 使 用 SQL 直接 编写 。 为 此 ， 要 执行 如 下 步 又 。 





(1) 在 Open Office Base 中 打开 数据 库 。 
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(2) 选择 左 侧 Database 面板 中 的 Queries, 

(3) 在 Tasks 面板 ， 点 击 Create Query In SQL View 以 显示 Query Design 
窗口 。 

(4) 在 大 文本 框 〈 整 个 窗口 都 是 文本 框 ) 中 输入 SQL 语句 。 

(5) 执行 SQL 语句 ， 点 击 Run Query 按钮 《有 文档 和 绿色 对 钩 的 按钮 )。 
也 可 以 按 F5 或 从 Edit 菜单 选择 Run Query， 执 行 SQL 语句 。 























B.2 使 用 Adobe ColdFusion 

Adobe ColdFusion 是 一 个 Web 应 用 开发 平台 。ColdFusion 使 用 一 个 基于 
标签 的 语言 来 创建 脚本 。 为 测试 SQL, 需要 创建 一 个 可 从 Web 浏览 器 调 
用 执行 的 简单 页 面 。 执 行 以 下 步骤 。 














(1) 在 从 ColdFusion 代码 内 使 用 任意 数据 库 之 前 ， 必 须 定 义 一 个 数据 源 。 
ColdFusion Administrator 程 序 提供 了 一 个 定义 数据 源 的 基于 Web 的 界 
HI (如 果 需 要 帮助 ， 请 参阅 ColdFusion 文档 )。 

(2) 创建 新 的 ColdFusion 页 (用 CFM 扩展 )。 

(3) 使 用 CFML <CFQUERY> 和 </CFQUERY> 标 签 创建 一 个 查询 块 。 用 NAME 

属性 命名 它 并 在 DATASOURCE 属性 中 定义 Data Source, 

(4) 在 <CFQUERY> 和 </CFQUERY> 标 签 之 间 输 入 SQL 语句 。 

(5) 使 用 <CFDUMP> 或 <CFOUTPUT> 循 环 显示 查询 结果 。 

(6) 在 Web 服务 器 根 目录 下 的 任意 可 执行 目录 中 保存 此 页 面 。 

(7) 通过 从 Web 浏览 器 中 调用 此 页 面 执行 它 。 



































B.3 使 用 IBM DB2 


IBM 的 DB2 是 一 个 强 有 力 的 高 端 多 平台 DBMS。 它 带 有 一 整套 可 用 来 执 











图 灵 社 区 会 员 maik000 专 享 尊重 版 权 














212 | 附录 B ”流行 的 应 用 程序 








ÍT SQL 语句 的 客户 端 工具 。 下 面 的 说 明 使 用 基于 Java 的 Control Center 
实用 程序 ， 因 为 它 是 最 简单 且 最 通用 的 绑 定 应 用 程序 。 


























(1) 运行 Control Center。 

(2) 左 侧 的 Object View 列 出 了 所 有 可 用 的 数据 库 ， 展 开 All Databases, 
选择 你 需要 的 数据 库 。 

(3) 在 选择 的 数据 库 上 点 击 右键 并 选择 Query， 或 者 〈 虽然 它 被 选中 ) 从 
Selected 菜单 选择 Query。 

(4) 在 上 面 的 框 中 输入 SQL 语句 。 

(5) 点 击 Execute 按钮 (有 绿色 朝 右 第 头 的 按钮 )， 执 行 这 个 脚本 。 

(6) 下 面 的 窗口 会 显示 状态 信息 , 切换 到 Query Results 标签 页 ， 可 以 以 网 
格 形式 显示 结果 。 




















B.4 使 用 MariaDB 


MariaDB 没有 自己 的 客户 端 实用 程序 ， 而 是 使 用 MySQL 的 客户 端 实用 
程序 (完全 兼容 )。 请 参阅 B.10 节 。 
































B.5 使 用 Microsoft Access 


Microsoft Access 通常 用 来 交互 式 地 创建 和 管理 数据 库 ， 并 且 交 互 式 地 处 
理 数 据 ，Access 给 出 Query Designer， 可 用 来 交互 式 地 建立 SQL 语句 。 
Query Designer 的 一 个 相当 有 用 的 特性 是 ， 它 也 允许 给 出 直接 执行 的 
SQL。 为 使 用 这 个 特性 ， 进 行 如 下 操作 。 





























(1) 运行 Microsoft Access。 它 将 提示 你 打开 (或 创建 ) 数据 库 。 打 开 要 使 
用 的 数据 库 。 
(2) 在 Database 窗 口中 选择 Queries, 然 后 单 击 New 按钮 并 选择 Design View, 
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(3) 出 现 Show Table 对 话 框 ,关闭 此 窗口 ， 不 选择 任何 表 。 

(4) 从 View 菜单 选择 SQL View 显示 Query 窗口 。 

(5) 在 Query 窗口 中 输入 要 执行 的 SQL 语句 。 

(6) 单 击 Run 按钮 (有 红 惊 叹 号 的 那个 ) 执行 SQL 语句 。 这 将 切换 视图 
到 Datasheet View ( 它 将 在 网 格 中 显示 结果 )。 

(D 根据 需要 在 SQL View 和 Datasheet View 间 切 换 (为 改动 SQL 语句 ， 
需 返 回 到 SQL View )。 还 可 以 使 用 Design View 交互 式 地 建立 SQL 
语句 。 














B.6 使 用 Microsoft ASP 

Microsoft ASP 是 创建 基于 Web 应 用 的 脚本 平台 。 在 ASP 页 内 测试 SQL 
语句 ， 必 须 创 建 一 个 通过 从 Web 浏览 器 调用 来 执行 的 页 面 。 下 面 是 在 
ASP 页 面 内 执行 SQL 语句 所 需 的 步 又。 


























(1) ASP {EJH ODBC 与 数据 库 交 互 ,因此 在 继续 进行 之 前 必须 给 出 ODBC 
数据 源 〈 请 参阅 本 附录 最 后 面 的 内 容 )。 

(2) 用 任意 文本 编辑 器 创建 新 ASP 页 面 ( 带 ASP 扩展 )。 

(3) 使 用 Server.Create0bject 创建 ADODB . Connection 对 象 的 一 个 
实例 。 

(4) 使 用 Open 方法 打开 所 需 的 ODBC 数据 源 。 

(5) 将 SQL 语句 传递 到 一 个 对 Execute 方法 的 调用 。Execute 方法 返回 
结果 集 。 使 用 Set 命令 将 返回 的 结果 保存 到 结果 集 。 

(6) 为 显示 结果 ， 应 该 对 检索 出 的 数据 应 用 <% Do While NOT EOF %> 
循环 。 

(7) 在 Web 服务 器 根 目 录 下 的 任意 可 执行 目录 中 保存 此 页 面 。 

(8) 通过 从 Web 浏览 器 调用 ， 执 行 此 页 面 。 
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B.7 使 用 Microsoft ASP.NET 

Microsoft ASP.NET 是 一 个 使 用 NET 框架 ， 创 建 基 于 Web 应 用 的 脚本 平 
台 。 在 ASPNET 页 面 内 测试 SQL 语句 ， 必 须 创 建 一 个 可 通过 从 浏览 器 
调用 来 执行 的 页 面 。 有 多 种 方法 完成 这 项 工作 , 下 面 是 其 中 的 一 种 方法 。 

















(1) 创建 一 个 带 .aspx 扩展 的 新 文件 。 

(2) 用 Sql1Connection0) 或 01eDbConnection() 创 建 数据 库 连 接 。 
(3) 使 用 Sq1Command () zk: 01eDbCommand O 2A DBMS 传递 语句 。 
(4) 用 ExecuteReader 创建 一 个 DataReader。 

(5) 对 返回 的 读 和 器 (reader ) 循环 以 获得 返回 值 。 

(6) 在 Web 服务 器 根 目录 下 的 任意 可 执行 目录 中 保存 页 面 。 

(7) 通过 从 Web 浏览 需 调 用 页 面 执行 它 。 








B.8 使 用 Microsoft Query 


Microsoft Query 是 一 个 独立 的 SQL 查询 工具 , 也 是 一 个 对 ODBC 数据 源 
测试 SQL 语句 的 理想 实用 程序 。 大 多 数 Windows 安装 程序 不 安装 
Microsoft Query， 不 过 ， 它 可 以 与 其 他 Microsoft 产品 以 及 第 三 方 产品 一 
起 选择 安装 。 如 果 你 的 计算 机 上 存在 就 可 以 使 用 。 



































提示 : 获得 MS-Query 

MS-Query 通常 与 Office 等 其 他 微软 产品 一 起 安装 , 虽然 它 只 能 在 进行 
完整 安装 时 安装 。 如 果 Start 按钮 上 没有 给 出 ， 可 用 Start Find 在 系统 
上 查找 它 (不管 你 是 否 知 道 ， 一 般 都 会 给 出 它 )。 要 查找 的 文件 为 
MSQRY32.EXE 或 MSQUERY.EXE。 











使 用 Microsoft Query， 进 行 如 下 操作 。 
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(1) Microsoft Query 使 用 ODBC 与 数据 库 打交道 , 因此 在 可 以 继续 进行 之 
前 必须 给 出 ODBC 数据 源 (参见 B.16 的 说 明 )。 

(2) 使 用 Microsoft Query 之 前 ， 必 须 在 计算 机 上 安装 它 。 请 浏览 Start 按 
钮 下 的 程序 组 找到 它 。 

(3) 从 File 菜单 中 ， 选 择 Execute SQL， 显 示 Execute SQL 窗 。 

(4) 单 击 Data Sources 按钮 选择 所 要 的 ODBC 数据 源 。 如 果 所 需 的 数据 源 
未 列 出 ， 单 击 Other 寻找 。 选 择 了 合适 的 数据 源 后 ， 单 击 Use 按钮 。 

(5) 在 SQL Statement 框 中 输入 SQL 语句 。 

(6) 单 击 Execute 执行 SQL 语句 并 显示 返回 数据 。 
































B.9 使 用 Microsoft SQL Server (包括 Microsoft 
SQL Server Express) 


Microsoft SQL Server 给 出 了 一 个 称 为 SQL Server Management Studio 的 强 
大 管理 工具 。 这 个 工具 用 来 管理 数据 库 、 管 理 安 全 、 制 作 报表 等 各 种 工 
作 ， 给 出 了 编写 和 测试 SQL 语句 的 理想 环境 。 以 下 说 明 如 何 使 用 SQL 


Server Management Studio。 























(1) 运行 SQL Server Management Studio。 

(2) 如 果 需 要 ， 提 供 服务 器 和 登录 信息 。 

(3) SQL Server Management Studio 会 展示 多 个 面板 。 左 侧 的 Object 
Explorer 列 出 了 所 有 的 数据 库 及 其 详细 信息 ， 上 面 的 工具 条 包含 功能 
按钮 ， 最 大 的 文本 区 域 用 于 输入 SQL 语句 。 

(4) 工具 条 下 方 左 侧 的 下 拉 选 择 框 显示 当前 的 数据 库 ， 也 人 允许 改变 数据 
库 。( 做 出 选择 在 功能 上 等 同 于 发 出 一 条 USE 语句 。) 确保 你 使 用 的 数 
据 库 是 正确 的 ， 如 果 需 要 就 切换 数据 库 。 
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(5) 在 文本 区 域 输入 SQL 语句 ， 然 后 点 击 Execute Query 按钮 (有 红色 感 
到 号 的 )， 执 行 它 (还 可 按 F5 或 从 Query 菜单 选择 Execute )。 

(6) 结果 将 显示 在 SQL 窗 下 的 独立 窗 格 中 。 

(7) 单 击 查询 屏幕 底部 的 标签 页 , 在 查看 数据 与 查看 返回 消息 和 信息 之 间 
切换 。 


























B.10 使 用 MySQL 


有 两 种 方式 使 用 MySQL. DBMS 带 有 一 个 名 为 mysql 的 命令 行 实用 程 
序 。 这 是 一 个 纯 文本 工具 ， 通 常 作为 MySQL 安装 程序 的 一 部 分 来 安装 ， 
用 来 执行 任何 SQL 语句 。 另 外 ,MySQL 的 创建 者 发 布 了 一 个 名 为 MySQL 
Workbench 的 交互 工具 ， 通 常 需要 独立 下 载 和 安装 ， 所 以 它 不 会 出 现在 
其 他 安装 程序 中 。 学 习 MySQL 时 ， 强 烈 推荐 使 用 它 。 





























从 命令 行使 用 mysq1， 进 行 如 下 操作 。 





(1) 输入 mysq1 运行 实用 程序 。 根 据 如 何 定义 安全 性 ， 可 能 需要 使 用 -u 
和 -p 参数 指定 登录 信息 。 

(2) 在 mysq1> 提 示 下 输入 USE database 打开 数据 库 ， 例 如 USE tysql 
就 是 打开 tysql 数据 库 。 

(3) 在 mysq1> 提 示 下 输入 SQL 语句 ， 每 条 语句 必须 以 分 号 ( ; ) 结束 。 结 
果 将 显示 在 屏幕 上 。 

(4) 为 可 能 使 用 的 命令 列表 输入 \h， 为 状态 信息 输入 \s (包括 MySQL 版 
本 信息 )。 

(5) 输入 \q 退出 mysql 实用 程序 。 




















使 用 MySQL Workbench， 进 行 如 下 操作 。 


(1) 运行 MySQL Workbench。 
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(2) MySQL Workbench 的 最 左 侧 列 出 了 可 用 的 MySQL 数据库 连 接 ， 允许 
你 访问 它们 。 点 击 任何 连接 就 可 以 打开 这 个 数据 库 ; 如 果 数 据 库 没有 
在 此 列 出 ， 请 选择 New Connection。 

(3) 一 旦 连接 ,就 会 显示 多 个 窗口 。 左 侧 的 Object Browser 列 出 了 可 用 的 数据 库 ， 
中 间 是 用 于 输入 SQL 语句 的 大 的 文本 编辑 器 ， 结 果 或 信息 显示 在 下 方 。 

(4) 点 击 + SQL 按钮 ， 打 开 新 的 SQL 窗口 。 

(5) 输入 SQL 语句 后 ， 点 击 Execute ( 有 闪电 图 片 的 那个 ) 运行 SQL。 结 
果 显 示 在 下 面 。 
































B.11 使 用 Oracle 

Oracle 有 一 大 套 工具 和 客户 端 。 学 习 SQL 时 首选 Oracle SQL Developer, 
它 可 以 与 MySQL DBMS 一 起 安装 ， 也 可 以 独立 下 载 并 安装 。 下 面 介绍 
如 何 使 用 此 工具 。 





(1) 运行 Oracle SQL Developer ( Windows 用 户 需 要 使 用 提供 的 批 处 理 文 
件 运行 ， 不 能 通过 应 用 本 里 运行 )。 

(2) 在 使 用 数据 库 前 , 你 需要 定义 一 个 连接 。 这 可 以 使 用 左 侧 Connections 
面板 中 的 选项 完成 。 

(3) 连接 完成 后 ,使 用 SQL Worksheet 标签 页 E Query Builder 屏幕 中 输 
入 SQL 语句 。 

(4) 执行 SQL 语句， 单 击 Execute 按钮 〈 带 内 电网 形 )。 结 果 将 显示 在 下 
面 的 面板 中 。 
































B.12 使 用 Oracle Express 
Oracle Express 是 一 种 强 有 力 、 易 于 使 用 的 DBMS ， 有 着 非常 直观 、 基 于 
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Web 的 用 户 界面 。 一 旦 安装 了 Oracle Express， 就 会 有 个 Getting Started 
来 启动 Web 页 面 管理 , 也 可 以 使 用 该 页 面 来 运行 SQL 命令 。 为 此 , 要 按 
以 下 步 又 操作 。 











(1) 运行 Oracle Express 管理 Web 页 面 。 

(2) 出 现 提示 ， 使 用 安装 时 的 用 户 名 和 密码 登录 。 

(3) 登录 后 可 以 看 到 一 系列 图 标 ， 包 括 一 个 含 单词 SQL 的 屏幕 图 片 。 点 
击 这 个 图 标 可 以 访问 选项 。 

(4) 第 一 个 图 标 是 SQL Commands, 可 用 来 输入 SQL 语句 。( 第 二 个 图 标 是 
SQL Scripts， 执 行 已 经 编写 的 SQL 脚本 时 很 有 用 ， 像 创建 和 填充 本 书 
可 用 的 脚本 。) 点 击 SQL Commands 图 标 ， 打 开 SQL Commands 窗口 。 

(5) 在 屏幕 上 方 输入 SQL 语句 。 

(6) 执行 SQL 查询 , 点 击 右上 方 的 Run 按钮 。 结 果 显 示 在 SQL 语句 之 下 。 


















































B.13 使 用 PHP 


PHP 是 一 个 流行 的 Web 脚本 语言 。 它 提供 了 用 来 连接 各 种 数据 库 的 函数 
和 库 ， 因 此 ， 用 来 执行 SQL 语句 的 代码 可 根据 所 用 (以 及 如 何 访问 ) 的 
DBMS 而 变化 。 因而 , 这 里 无 法 提供 用 于 各 种 DBMS 和 各 种 情形 的 步骤 ， 
只 提供 一 个 使 用 MySQL 的 一 般 例 子 。 关 于 如 何 连接 到 具体 的 DBMS 的 
介绍 ， 请 参阅 PHP 文档 。 














(1) 创建 一 个 新 的 PHP 页 面 (和 一 个 PHP 扩展 )。 

(2) 使 用 适当 的 函数 连接 数据 库 。 使 用 mysql_connect O 12 MySQL 数 
据 库 。 

(3) 将 SQL 语句 传 给 适当 的 查询 函数 。 针对 MySQL, 使 用 mysq1_queryQ)。 

(4) 返回 一 组 结果 ,循环 遍历 它们 以 显示 。 
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(5) 将 页 面 保存 在 Web 服务 器 根 目 录 下 的 任何 可 执行 目录 中 。 
(6) 通过 从 Web 浏览 需 调 用 来 执行 页 面 。 


B.14 使 用 PostgreSQL 


有 两 种 方式 使 用 PostgreSQL。DBMS 带 有 一 个 名 为 psq1 的 命令 行 实用 
程序 ， 这 是 一 个 纯 文 本 工具 ,通常 与 PostgreSQL 一 起 安装 ， 用 来 执行 任 
意 SQL 语句 。 另 外 , 有 一 个 名 为 pgAdmin 的 交互 工具 , 主要 用 于 DBMS 
管理 ， 可 测试 SQL 语句 。 
































使 用 psq1 ， 进 行 如 下 操作 。 








(1) 输入 psq1 运行 此 实用 程序 。 加 载 一 个 特定 的 数据 库 ， 请 在 命令 行 上 
指定 它 ， 格 式 为 : psq1 database ( PostgreSQL 不 支持 USE 命令 )。 
(2) 在 => 提 示 下 输入 要 执行 的 SQL 语句 ， 每 条 语句 以 分 号 ( ; ) 结束 。 结 
果 将 显示 在 屏幕 上 。 

(3) 输入 \?， 可 列 出 可 能 用 到 的 命令 。 

(4) 输入 \h， 可 得 到 SQL 帮助 ; 输入 \h statement (如 \h SELECT), 
可 得 到 特定 SQL 语句 的 帮助 。 

(5) 输入 \q 退出 psq1 实用 程序 。 








使 用 pgAdmin， 进 行 如 下 操作 。 


(1) 运行 pgAdmin ( 它 也 可 能 称 为 pgAdmin II )。 

(2) 如 果 出 现 提 示 ， 提 供 登录 信息 。 

(3)pgAdmin 会 在 左 侧 列 出 数据 库 服务 器 。 选 择 要 使 用 的 数据 库 ,pgAdmin 
将 连接 它 ， 并 显示 服务 器 信息 。 

(4) 然后 选择 实际 的 数据 库 ( 这 样 做 ,工具 栏 按钮 即 可 使 用 了 )。 
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(5) 定位 Execute Arbitrary SQL Queries 按钮 ( 是 的 ， 这 确实 是 它 的 名 字 ， 
它 是 那个 包含 单词 SQL 的 放大 镜 图 片 )。 

(6) 将 显示 一 个 新 的 屏幕 。 确 保 要 用 的 数据 库 列 在 了 右上 方 的 下 拉 选 择 
框 中 。 

(7) 现在 可 以 在 屏幕 上 方 的 大 文本 框 中 输入 SQL 语句 了 。 

(8) 点 击 Execute Query 按钮 (有 绿色 向 右 箭头 的 那个 )， 执 行 SQL 语句 。 
结果 显示 在 下 面 。 














B.15 使 用 SQLite 


SQLite 通常 在 其 他 语言 和 应 用 程序 中 使 用 ， 不 作为 单独 的 数据 库 使 用 。 
然而 ，SQLite 确实 有 一 个 命令 行 实 用 程序 ， 可 以 用 来 执行 SQLite 数据 库 
的 语句 。SQLite 命令 行 实用 程序 称 为 sqlite3 (或 Windows 上 的 
sqlite3.exe )。 从 命令 行使 用 sqlite3， 要 进行 如 下 操作 。 














(1) 理想 情况 下 ，sq1ite3 和 数据 库 可 以 放 在 同一 文件 夹 下 。( SQLite 数 
据 库 包含 在 有 .sqlite 或 .db 扩展 名 的 单独 文件 中 , 但 事实 上 可 以 是 任意 
扩展 名 ， 或 者 没有 扩展 名 。) 

(2) 输入 sqlite3 database.sqlite ( 用 实际 的 数据 库 文件 名 替换 
database.sqlite )。 

(3) 可 以 看 到 sqlite> 提 示 符 ， 在 此 输入 SQL 语句 。 所 有 语句 必须 用 分 
号 (; ) 结束 。 

(4) 默认 情况 下 ，sq1ite3 用 列 之 间 的 管道 字符 显示 返回 的 数据 ,， 且 没有 
标题 。 要 改变 这 种 行为 ， 输 入 .mode colum 并 按 回 车 ， 然 后 输 
入 .header on 并 按 回 车 。 

(5) 输入 .quit 并 按 回 车 ， 退 出 sqlite3, 
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B.16 配置 DDBC 数 据 源 


上 面 描述 的 几 个 应 用 程序 使 用 了 ODBC 进行 数据 库 集成 ， 因 此 ， 这 里 简 
要 概述 一 下 ODBC， 以 及 配置 ODBC 数据 源 的 指令 。 

















ODBC 是 一 个 标准 ， 能 使 客户 端 应 用 与 不 同 的 后 端 数据 库 或 基础 数据 库 
引 警 交互。 使 用 ODBC， 能 够 在 一 个 客户 端 中 编写 代码 ， 并 使 前 述 各 种 
工具 与 几乎 所 有 数据 库 或 DBMS 交互 。 

















ODBC 本 身 不 是 数据 库 ， 但 它 包 装 了 数据 库 ， 使 所 有 数据 库 以 一 致 和 清 
晰 定义 的 方式 工作 。 它 利用 具有 两 种 主要 功能 的 软件 驱动 程序 实现 这 一 
点 。 首 先 ， 它 们 封装 数据 库 的 本 身 特 性 或 特色 ， 并 对 客户 端 隐藏 它们 。 
其 次 ,它们 提供 一 种 常用 的 语言 与 这 些 数 据 库 交互 ( 在 需要 时 进行 转换 )。 
ODBC 所 用 的 语言 就 是 SQL。 



































ODBC 客户 端 应 用 程序 并 不 直接 与 数据 库 交 互 ， 而 是 与 ODBC 数据 源 
交互 。 数 据 源 是 一 个 逻辑 数据 库 ， 包 括 驱 动 程序 ( 每 种 类 型 的 数据 库 
有 自己 的 驱动 程序 ) 和 如 何 连接 到 数据 库 的 信息 (文件 路 径 、 服 务 器 
名 等 )。 











定义 了 ODBC 数据 源 后 , 任何 兼容 ODBC 的 应 用 程序 都 可 以 使 用 这 些 数 
据 源 。ODBC 数据 源 并 不 针对 具体 的 应 用 程序 ， 它 们 针对 的 是 系统 。 

















注意 : ODBC 的 差别 
存在 许多 不 同 的 ODBC 程序 版 本 ， 因 此 不 可 能 提供 适用 于 所 有 版 本 的 
Ao 在 设置 具体 的 数据 源 时 ， 应 该 密切 注意 具体 的 提示 。 











ODBC 数 据 源 用 Windows ControlPanel 的 ODBC 程 序 来 定义 。 设 置 ODBC 
数据 源 ， 进 行 如 下 操作 。 
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(1) 打开 Windows Control Panel 的 ODBC 程序 。 

(2) 大 多 数 ODBC 数据 源 应 该 设置 为 系统 范围 的 数据 源 ( 相对 于 用 户 专用 
的 数据 源 )， 因 此 ， 如 果 可 以 ， 应 该 选择 System DSN。 

(3) 单 击 Add 按钮 添加 新 的 数据 源 。 

(4) 选择 要 使 用 的 驱动 程序 。 通 常 有 一 组 默认 的 驱动 程序 ,支持 主要 的 微 
软 产 品 。 你 的 系统 上 也 可 以 安装 其 他 驱动 程序 。 必 须 选 择 一 个 与 将 要 
连接 到 的 数据 库 类 型 相 匹配 的 驱动 程序 。 

(5) 系统 根据 数据 库 或 DBMS 的 类 型 ， 提 示 输 入 服务 器 名 或 文件 路 径 信 
息 ， 以 及 可 能 的 登录 信息 。 根 据 要 求 提 供 这 些 信 息 ， 然 后 遵循 其 他 提 
示 创 建 数据 源 。 
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附录 C SQL 语句 的 语法 


为 帮助 读者 在 需要 时 找到 相应 语句 的 语法 ， 本 附录 列 出 了 最 常 使 用 的 
SQL 语句 的 语法 。 每 条 语句 以 简要 的 描述 开始 ， 然 后 给 出 它 的 语法 。 为 
更 方便 查询 ， 还 标注 了 相应 语句 所 在 的 课 。 








在 阅读 语句 语法 时 ， 应 该 记 住 以 下 约定 。 





a | 符号 用 来 指出 几 个 选择 中 的 一 个 ， 因 此 ，NULL | NOT NULL 表示 或 
者 给 出 NULL 或 者 给 出 NOT NULL。 

口 包含 在 方 括号 中 的 关键 字 或 子 句 (如 [1ike this] ) 是 可 选 的 。 

口 下 面 列 出 的 语法 几乎 对 所 有 DBMS 都 有 效 。 关 于 具体 语法 可 能 变动 的 
细节 ， 建 议 读者 参考 自己 的 DBMS 文档 。 


























C.1 ALTER TABLE 





ALTER TABLE 用 来 更 新 已 存在 表 的 结构 。 为 了 创建 新 表 ， 应 该 使 用 
CREATE TABLE。 详 细 信 息 ， 请 参阅 第 17 课 。 





输入 了 


ALTER TABLE tablename 

( 
ADD | DROP column datatype [NULL |NOT NULL] [CONSTRAINTS], 
ADD | DROP column datatype [NULL |NOT NULL] [CONSTRAINTS], 
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); 


C.2 COMMIT 


COMMIT 用 来 将 事务 写 和 数据库。 详细 内 容 请 参阅 第 20 课 。 








输入 了 


COMMIT [TRANSACTION] ; 


C.3 CREATE INDEX 





CREATE INDEX 用 于 在 一 个 或 多 个 列 上 创建 索引 。 详 细 内 容 请 参阅 第 
22 课 。 


输入 了 


CREATE INDEX indexname 
ON tablename (column, ...); 


C.4 CREATE PROCEDURE 








CREATE PROCEDURE 用 于 创建 存储 过 程 。 详 细 内 容 请 参阅 第 19 课 。 正 如 
所 述 ，Oracle 使 用 的 语法 稍 有 不 同 。 





输入 
CREATE PROCEDURE procedurename [parameters] [options] 


AS 
SQL statement; 


灵 社 区 会 员 maik000 专 享 尊重 版 权 





附录 C SOL 语句 的 语法 | 225 
C.5 CREATE TABLE 


CREATE TABLE 用 于 创建 新 数据 库 表 。 更 新 已 经 存在 的 表 的 结构 ， 使 用 
ALTER _ TABLE。 详细 内 容 请 参阅 第 17 课 。 








输入 了 


CREATE TABLE tablename 
( 


column datatype [NULLINOT NULL] [CONSTRAINTS], 
column datatype [NULLINOT NULL] [CONSTRAINTS], 


C.6 CREATE VIEW 


CREATE VIEW 用 来 创建 一 个 或 多 个 表 上 的 新 视图 。 详细 内 容 请 参阅 第 18 课 。 








输入 了 


CREATE VIEW viewname AS 
SELECT columns, ... 
FROM tables, ... 

[WHERE ...] 

[GROUP BY ...] 

[HAVING ...]; 


C.7 DELETE 
DELETE 从 表 中 删除 一 行 或 多 行 。 详 细 内 容 请 参阅 第 16 课 。 





输入 了 


DELETE FROM tablename 
[WHERE ...]; 
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C.8 DROP 


DROP 永久 地 删除 数据 库 对 象 ( 表 、 视 图 、 索 引 等 ), 详细 内 容 请 参阅 第 17、 
18 课 。 





输入 了 


DROP INDEX|PROCEDURE|TABLE|VIEW 
indexname |procedurename|tablename |viewname; 


C.9 INSERT 


INSERT 为 表 添 加 一 行 。 详 细 内 容 请 参阅 第 15 课 。 


输入 
INSERT INTO tablename [(columns, ...)] 
VALUES(values, ...); 


C.10 INSERT SELECT 


INSERT SELECT 将 SELECT 的 结果 插 和 人 到 一 个 表 。 详 细 内 容 请 参阅 多 
15 Wo 


dr 





输入 

INSERT INTO tablename [(columns, ...)] 
SELECT columns, ... FROM tablename, ... 
[WHERE ...]; 


C.11 ROLLBACK 
ROLLBACK 用 于 撤销 一 个 事务 块 。 详 细 内 容 请 参阅 第 20 课 。 
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输入 
ROLLBACK [ TO savepointname] ; 
或 者 : 


输入 了 


ROLLBACK TRANSACTION; 


C.12 SELECT 


SELECT 用 于 从 一 个 或 多 个 表 (视图 ) 中 检索 数据 。 更 多 的 基本 信息 ,请 
参阅 第 2、3、4 课 (2 ~ 14 课 都 与 SELECT 有 关 )。 





输入 了 


SELECT columnname, ... 
FROM tablename, ... 


[WHERE ...] 
[UNION ...] 
[GROUP BY ...] 
[HAVING ...] 


[ORDER BY ...]; 


C.13 UPDATE 


UPDATE 更 新 表 中 的 一 行 或 多 行 。 详 细 内 容 请 参阅 第 16 课 。 





输入 
UPDATE tablename 


SET columname = value, ... 
[WHERE ...]; 
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正如 第 1 课 所 述 ， 数 据 类 型 是 定义 列 中 可 以 存储 什么 数据 以 及 该 数据 实 
际 怎样 存储 的 基本 规则 。 





数据 类 型 用 于 以 下 目的 。 





口 数据 类 型 允许 限制 可 存储 在 列 中 的 数据 。 例 如 ， 数 值 数 据 类 型 列 只 能 
接受 数值 。 

口 数据 类 型 允许 在 内 部 更 有 效 地 存储 数据 。 可 以 用 一 种 比 文本 字符 串 更 
简洁 的 格式 存储 数值 和 日 期 时 间 值 。 

O 数据 类 型 允许 变换 排序 顺序 。 如 果 所 有 数据 都 作为 字符 串 处 理 ， 则 
1 位 于 10 之 前 ,而 10 又 位 于 2 之 前 (字符 串 以 字典 顺序 排序 ， 从 
左边 开始 比较 , 一 次 一 个 字符 )。 作 为 数值 数据 类 型 , 数值 才能 正确 
排序 。 
































在 设计 表 时 ， 应 该 特别 重视 所 用 的 数据 类 型 。 使 用 错误 的 数据 类 型 可 能 
会 严重 影响 应 用 程序 的 功能 和 性 能 。 更改 包 含 数据 的 列 不 是 一 件 小 事 ( 而 
且 这 样 做 可 能 会 导致 数据 丢失 )。 





























本 附录 虽然 不 是 关于 数据 类 型 及 其 如 何 使 用 的 完整 教材 ， 但 介绍 了 主要 
的 数据 类 型 、 用 途 、 兼 容 性 等 问题 。 
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注意 : 任意 两 个 DBMS 都 不 是 完全 相同 的 

以 前 曾经 说 过 ， 现 在 还 需要 再 次 提醒 。 不 同 DBMS 的 数据 类 型 可 能 
很 大 的 不 同 。 在 不 同 DBMS 中 ,即使 具有 相同 名 称 的 数据 类 型 也 可 能 
代表 不 同 的 东西 。 关 于 具体 的 DBMS 支持 何 种 数据 类 型 以 及 如 何 支持 
的 详细 信息 ， 请 参阅 具体 的 DBMS 文档 。 








D.1 字符 串 数 据 类 型 
最 常用 的 数据 类 型 是 字符 串 数据 类 型 。 它 们 存储 字符 串 ， 如 名 字 、 地 址 、 
电话 号 码 、 邮 政 编码 等 。 有 两 种 基本 的 字符 串 类 型 ， 分 别 为 定 长 字符 串 
和 变 长 字符 串 (ZILK D-1) 

































































定 长 字符 串 接受 长 度 固定 的 字符 串 , 其 长 度 是 在 创建 表 时 指定 的 。 例 如 ， 
名 字 列 可 允许 30 个 字符 ， 而 社会 安全 号 列 允 许 11 个 字符 〈 允许 的 字符 
数目 中 包括 两 个 破 折 号 )。 定 长 列 不 允许 多 于 指定 的 字符 数目 。 它 们 分 配 
的 存储 空间 与 指定 的 一 样 多 。 因 此 , 如 果 字 符 串 Ben 存储 到 30 个 字符 的 
名 字 字 段 ， 则 存储 的 是 30 个 字符 , 缺少 的 字符 用 空格 填充 , 或 根据 需要 
补 为 NULL。 

















变 长 字符 串 存 储 任意 长 度 的 文本 〈 其 最 大 长 度 随 不 同 的 数据 类 型 和 
DBMS 而 变化 ) 有 些 变 长 数据 类 型 具有 最 小 的 定 长 , 而 有 些 则 是 完全 变 
长 的 。 不 管 是 哪 种 ， 只 有 指定 的 数据 得 以 保存 ( 额外 的 数据 不 保存 )。 
































既然 变 长 数据 类 型 这 样 灵活 ， 为 什么 还 要 使 用 定 长 数据 类 型 ? 答案 是 性 
能 。DBMS 处 理 定 长 列 远 比 处 理 变 长 列 快 得 多 。 此 外 ,许多 DBMS 不 允 
许 对 变 长 列 (或 一 个 列 的 可 变 部 分 ) 进行 索引 ， 这 也 会 极 大 地 影响 性 能 
(详细 请 参阅 第 22 课 )。 
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表 D-1 串 数据 类 型 
数据 类 型 说 明 
CHAR 1 ~ 255 个 字符 的 定 长 字符 串 。 它 的 长 度 必须 在 创建 时 规定 
NCHAR CHAR 的 特殊 形式 ， 用 来 支持 多 字 节 或 Unicode 字 符 ( 此 类 型 
的 不 同 实现 变化 很 大 ) 
NVARCHAR TEXT 的 特殊 形式 ， 用 来 支持 多 字 节 或 Unicode 字 符 (此 类 型 





























的 不 同 实现 变化 很 大 ) 


TEXT( 也 称 为 LONG、MEMO ZKA 
或 VARCHAR ) 





提示 : 使 用 引号 
不 管 使 用 何 种 形式 的 字符 串 数据 类 型 ,字符 串 值 都 必须 括 在 单 引 号 内 。 








注意 : 当 数值 不 是 数值 时 

你 可 能 会 认为 电话 号 码 和 邮政 编码 应 该 存储 在 数值 字段 中 (数值 字段 
只 存储 数值 数据 ), 但 是 这 样 做 并 不 可 取 。 如 果 在 数值 字段 中 存储 邮政 
编码 01234， 则 保存 的 将 是 数值 1234， 实 际 上 丢失 了 一 位 数字 。 


需要 遵守 的 基本 规则 是 ， 如 果 数值 是 计算 (RE, PAF) 中 使 用 的 
数值 ， 则 应 该 存储 在 数值 数据 类 型 列 中 ; 如 果 作为 字符 事 ( 可 能 只 包 
ERF) 使 用 ， 则 应 该 保存 在 字符 囊 数据 类 型 列 中 。 











D.2 ”数值 数据 类 型 

数值 数据 类 型 存储 数值 。 多 数 DBMS 支持 多 种 数值 数据 类 型 ， 每 种 存储 
的 数值 具有 不 同 的 取 值 范围 。 显 然 ， 支 持 的 取 值 范 围 越 大 ， 所 需 存 储 空 
间 越 多 。 此 外 ， 有 的 数值 数据 类 型 支持 使 用 十 进 制 小 数 点 ( 和 小 数 )， 而 
有 的 则 只 支持 整数 。 表 D-2 列 出 了 常用 的 数值 数据 类 型 ,并 非 所 有 DBMS 
都 支持 所 列 出 的 名 称 约定 和 描述 。 















































TN 
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表 D-2 ”数值 数据 类 型 


















































数据 类 型 说 BB 
BIT 单个 二 进 制 位 值 ， 或 者 为 0 或 者 为 1 ， 主 要 用 于 开 / 关 标志 
DECIMAL ( 或 NUMERIC ) 定点 或 精度 可 变 的 浮 点 值 
FLOAT ( 或 NUMBER ) 浮 点 值 
INT ( 或 INTEGER ) 4 字 节 整数 值 ， 支 持 -2147483648 ~ 2147483647 的 数 
REAL 4 字 节 浮 点 值 
SMALLINT 2 字 节 整数 值 ， 支 持 -32768 ~ 32767 的 数 
TINYINT 1 字 节 整数 值 ， 支 持 0 ~ 255 的 数 




















提示 : 不 使 用 引号 
与 字符 串 不 一 样 ， 数 值 不 应 该 括 在 引号 内 。 








提示 : 货币 数据 类 型 

多 数 DBMS 支持 一 种 用 来 存储 货币 值 的 特殊 数值 数据 类 型 。 一 般 记 为 
MONEY 或 CURRENCY ， 这 些 数据 类 型 基本 上 有 是 有 特定 取 值 范围 的 
DECIMAL 数据 类 型 ， 更 适合 存储 货币 值 。 








D.3 ”日 期 和 时 间 数 据 类 型 


所 有 DBMS 都 支持 用 来 存储 日 期 和 时 间 值 的 数据 类 型 ( 见 表 D-3 )。 与 数值 
一 样 , 多 数 DBMS 都 支持 多 种 数据 类 型 , 每 种 具有 不 同 的 取 值 范围 和 精度 。 






































注意 : 指定 日 期 

不 存在 所 有 DBMS 都 理解 的 定义 日 期 的 标准 方法 。 多数 实现 都 理解 诸 
如 2015-12-30 或 Dec 30th, 2015 等 格式 , 但 即使 这 样 , 有 的 DBMS 
还 是 不 理解 它们 。 至 于 有 具体 的 DBMS 能 识别 哪些 日 期 格式 , 请 参阅 相 
应 的 文档 。 
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表 D-3 “日 期 和 时 间 数 据 类 型 























数据 类 型 说 BB 
DATE 日 期 值 
DATETIME ( 或 TIMESTAMP ) 日 期 时 间 值 
SMALLDATETIME 日 期 时 间 值 ， 精 确 到 分 ( 无 秒 或 毫秒 ) 
TIME 时 间 值 





提示 : ODBC 日 期 

因为 每 种 DBMS 都 有 自己 特定 的 日 期 格式 ,所 以 ODBC 创建 了 一 种 自 
己 的 格式 ,在 使 用 ODBC 时 对 每 种 数据 库 都 起 作用 。ODBC 格式 对 于 
日 期 类 似 于 {d '2005-12-30'}， 对 于 时 间 类 似 于 {t '21:46:29'}, 
而 对 于 日 期 时 间 类 似 于 {ts '2005-12-30 21:46:29'}。 如 果 通 过 
ODBC 使 用 SQL， 应 该 以 这 种 方式 格式 化 日 期 和 时 间 。 











D.4 ”二进制 数据 类 型 

二 进 制 数 据 类 型 是 最 不 具有 兼容 性 (幸运 的 是 ， 也 是 最 少 使 用 ) 的 数据 
类 型 。 与 迄今 为 止 介绍 的 所 有 数据 类 型 ( 它们 具有 特定 的 用 途 ) 不 一 样 ， 
二 进 制 数据 类 型 可 包含 任何 数据 ， 甚 至 可 包含 二 进 制 信息 ， 如 图 像 、 多 
媒体 、 字 处 理 文档 等 (参见 表 D-4 )。 
























































表 D-4 二进制 数据 类 型 






































数据 类 型 说 W 
BINARY 定 长 二 进 制 数据 ( 最 大 长 度 从 255B 到 8000B， 有 赖 于 具 
体 的 实现 ) 
a 变 长 二 进 制 数据 ， 最 长 2 GB 
RAW ( 某 些 实现 为 BINARY ) 定 长 二 进 制 数据 ， 最 多 255B 
VARB IPARI 变 长 二 进 制 数据 ( 最 大 长 度 一 般 在 255B 到 8000B 间 变 
化 ,依赖 于 具体 的 实现 ) 
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说 明 : 数据 类 型 对 比 

如 果 你 想 看 一 个 数据 库 比 较 的 实际 例子 ， 请 考虑 本 书 中 用 来 建立 样 例 
表 的 表 创建 脚本 (参看 附录 A )。 通 过 比较 这 些 用 于 不 同 DBMS 的 脚 
本 ， 可 看 到 数据 类 型 匹配 是 一 项 多 么 复杂 的 任务 。 
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MRE SQL 保留 字 




















SQL 是 由 关键 字 组 成 的 语言 , 关键 字 是 一 些 用 于 执行 SQL 操作 的 特殊 词 
汇 。 在 命名 数据 库 、 表 、 列 和 其 他 数据 库 对 象 时 ， 一 定 不 要 使 用 这 些 关 
键 字 ° 因此 9 这 些 关键 字 是 一 定 要 保留 的 ° 





























本 附录 列 出 主要 DBMS 中 最 常用 的 保留 字 。 请 注意 以 下 几 点 。 





口 关键 字 随 不 同 的 DBMS 而 变化 ， 并 非 下 面 的 所 有 关键 字 都 被 所 有 
DBMS 采用 。 

口 许多 DBMS 扩展 了 SQL 保留 字 , 使 其 包含 专门 用 于 实现 的 术语 。 多数 
DBMS 专用 的 关键 字 未 列 在 下 面 。 

口 为 保证 以 后 的 兼容 性 和 可 移植 性 ， 应 避免 使 用 这 些 保留 字 ， 即 使 它们 
不 是 你 使 用 的 DBMS 的 保留 字 。 


























ABORT AS BETWEEN 
ABSOLUTE ASC BIGINT 
ACTION ASCENDING BINARY 
ACTIVE ASSERTION BIT 

ADD AT BLOB 
AFTER AUTHORIZATION BOOLEAN 
ALL AUTO BOTH 
ALLOCATE AUTO-INCREMENT BREAK 
ALTER AUTOINC BROWSE 
ANALYZE AVG BULK 
AND BACKUP BY 

ANY BEFORE BYTES 
ARE BEGIN CACHE 
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CALL 
CASCADE 
CASCADED 
CASE 

CAST 
CATALOG 
CHANGE 
CHAR 
CHARACTER 
CHARACTER_LENGTH 
CHECK 
CHECKPOINT 
CLOSE 
CLUSTER 
CLUSTERED 
COALESCE 
COLLATE 
COLUMN 
COLUMNS 
COMMENT 
COMMIT 
COMMITTED 
COMPUTE 
COMPUTED 
CONDITIONAL 
CONFIRM 
CONNECT 
CONNECTION 
CONSTRAINT 
CONSTRAINTS 
CONTAINING 
CONTAINS 
CONTAINSTABLE 
CONTINUE 
CONTROLROW 
CONVERT 
COPY 

COUNT 
CREATE 
CROSS 
CSTRING 
CUBE 


CURRENT 
CURRENT_DATE 
CURRENT_TIME 
CURRENT_TIMESTAMP 
CURRENT_USER 
CURSOR 
DATABASE 
DATABASES 
DATE 
DATETIME 
DAY 

DBCC 
DEALLOCATE 
DEBUG 

DEC 

DECIMAL 
DECLARE 
DEFAULT 
DELETE 

DENY 

DESC 
DESCENDING 
DESCRIBE 
DISCONNECT 
DISK 
DISTINCT 
DISTRIBUTED 
DIV 

DO 

DOMAIN 
DOUBLE 

DROP 

DUMMY 

DUMP 

ELSE 

ELSEIF 
ENCLOSED 
END 

ERRLVL 
ERROREXIT 
ESCAPE 
ESCAPED 
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EXCEPT 
EXCEPTION 
EXEC 
EXECUTE 
EXISTS 
EXIT 
EXPLAIN 
EXTEND 
EXTERNAL 
EXTRACT 
FALSE 
FETCH 
FIELD 
FIELDS 
FILE 
FILLFACTOR 
FILTER 
FLOAT 
FLOPPY 
FOR 
FORCE 
FOREIGN 
FOUND 
FREETEXT 
FREETEXTTABLE 
FROM 
FULL 
FUNCTION 
GENERATOR 
GET 
GLOBAL 
GO 

GOTO 
GRANT 
GROUP 
HAVING 
HOLDLOCK 
HOUR 
IDENTITY 
IF 

IN 
INACTIVE 
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INDEX 
INDICATOR 
INFILE 
INNER 
INOUT 
INPUT 
INSENSITIVE 
INSERT 
INT 
INTEGER 
INTERSECT 
INTERVAL 
INTO 

IS 
ISOLATION 
JOIN 

KEY 

KILL 
LANGUAGE 
LAST 
LEADING 
LEFT 
LENGTH 
LEVEL 
LIKE 
LIMIT 
LINENO 
LINES 
LISTEN 
LOAD 
LOCAL 
LOCK 
LOGFILE 
LONG 
LOWER 
MANUAL 
MATCH 
MAX 
MERGE 
MESSAGE 
MIN 
MINUTE 


MIRROREXIT 
MODULE 
MONEY 
MONTH 
MOVE 
NAMES 
NATIONAL 
NATURAL 
NCHAR 
NEXT 

NEW 

NO 
NOCHECK 
NONCLUSTERED 
NONE 

NOT 

NULL 
NULLIF 
NUMERIC 
OF 

OFF 
OFFSET 
OFFSETS 
ON 

ONCE 
ONLY 
OPEN 
OPTION 
OR 

ORDER 
OUTER 
OUTPUT 
OVER 
OVERFLOW 
OVERLAPS 
PAD 

PAGE 
PAGES 
PARAMETER 
PARTIAL 
PASSWORD 
PERCENT 


PERM 
PERMANENT 
PIPE 

PLAN 
POSITION 
PRECISION 
PREPARE 
PRIMARY 
PRINT 
PRIOR 
PRIVILEGES 
PROC 
PROCEDURE 
PROCESSEXIT 
PROTECTED 
PUBLIC 
PURGE 
RAISERROR 
READ 
READTEXT 
REAL 
REFERENCES 
REGEXP 
RELATIVE 
RENAME 
REPEAT 
REPLACE 
REPLICATION 
REQUIRE 
RESERV 
RESERVING 
RESET 
RESTORE 
RESTRICT 
RETAIN 
RETURN 
RETURNS 
REVOKE 
RIGHT 
ROLLBACK 
ROLLUP 
ROWCOUNT 
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RULE 

SAVE 
SAVEPOINT 
SCHEMA 
SECOND 
SECTION 
SEGMENT 
SELECT 
SENSITIVE 
SEPARATOR 
SEQUENCE 
SESSION_USER 
SET 
SETUSER 
SHADOW 
SHARED 
SHOW 
SHUTDOWN 
SINGULAR 
SIZE 
SMALLINT 
SNAPSHOT 
SOME 

SORT 
SPACE 

SQL 
SQLCODE 
SQLERROR 
STABILITY 


STARTING 
STARTS 
STATISTICS 
SUBSTRING 
SUM 
SUSPEND 
TABLE 
TABLES 
TAPE 

TEMP 
TEMPORARY 
TEXT 
TEXTSIZE 
THEN 

TIME 
TIMESTAMP 
TO 

TOP 
TRAILING 
TRAN 
TRANSACTION 
TRANSLATE 
TRIGGER 
TRIM 

TRUE 
TRUNCATE 
UNCOMMITTED 
UNION 
UNIQUE 
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UNTIL 
UPDATE 
UPDATETEXT 
UPPER 
USAGE 
USE 

USER 
USING 
VALUE 
VALUES 
VARCHAR 
VARIABLE 
VARYING 
VERBOSE 
VIEW 
VOLUME 
WAIT 
WAITFOR 
WHEN 
WHERE 
WHILE 
WITH 
WORK 
WRITE 
WRITETEXT 
XOR 
YEAR 
ZONE 
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常用 SQL 语句 速 查 


ALTER TABLE 
ALTER TABLE 用 来 更 新 现存 表 的 模式 。 可 以 用 CREATE TABLE 来 创建 一 
个 新 表 。 详 情 可 参见 第 17 课 。 








COMMIT 
COMMIT 用 来 将 事务 写 和 数据库 。 详 情 可 参见 第 20 课 。 





CREATE INDEX 
CREATE INDEX 用 来 为 一 列 或 多 列 创建 索引 。 详 情 可 参见 第 22 课 。 








CREATE TABLE 
CREATE TABLE 用 来 创建 新 的 数据 库 表 。 可 以 用 ALTER TABLE 来 更 新 一 
个 现存 表 的 模式 。 详 情 可 参见 第 17 课 。 














CREATE VIEW 
CREATE VIEW 用 来 创建 一 个 或 多 个 表 的 视图 。 详 情 可 参见 第 18 课 。 


DELETE 
DELETE 用 来 从 表 中 删除 一 行 或 多 行 。 详 情 可 参见 第 16 课 。 








DROP 
DROP 用 来 永久 性 地 删除 数据 库 对 象 ( 表 、 视 图 、 索 引 等 )。 详 情 可 参见 
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第 17 课 和 第 18 课 。 





INSERT 
INSERT 用 来 对 表 添 加 一 个 新 行 。 详 情 可 参见 第 15 课 。 


INSERT SELECT 
INSERT SELECT 用 来 将 SELECT 的 结果 插入 到 表 中 。 详 情 可 参见 第 15 课 。 








ROLLBACK 
ROLLBACK 用 来 撤销 事务 块 。 详 情 可 参见 第 20 课 。 


SELECT 
SELECT 用 来 从 一 个 或 多 个 表 ( 或 视图 ) 中 检索 数据 。 详情 可 参见 第 2 课 、 
第 3 课 和 第 4 课 (第 2 课 到 第 14 课 从 不 同方 面 涉 及 了 SELECT )。 





UPDATE 
UPDATE 用 来 对 表 中 的 一 行 或 多 行进 行 更 新 。 详 情 可 参见 第 16 课 。 
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操作 符 ，37 
又 联结 ，107 
查询 ，93 
触发 器 ，199 
存储 过 程 ，166 
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导出 列 ，61 
等 值 联结 ，108 
笛 卡 儿 积 ，105 


F 





分 组 ，85 


去 
EN 











可 伸缩 ，103 
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提交 ，177 
通配符 ，14，46 
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外 键 ，142，193 


外 联结 ，116 
完全 限定 列 名 ， 
100 

唯一 约束 ，194 
谓词 ，47 
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引号 ，33 

引用 完整 性 ，190 
隐 式 提交 ，179 
游标 ，183 
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SQL 
必 知 必 会 
(第 4 版 ) 


SQL 语法 简洁 ， 使 用 方式 灵活 ， 功 能 强大 ， 已 经 成 为 当今 程序 员 不 可 或 缺 的 技能 。 

本 书 是 深 受 世界 各 地 读者 欢迎 的 SQL 经 典 畅 销 书 ， 内 容 丰 富 ， 文 字 简 洁 明 快 ， 针 对 Oracle、 
SOL Server、MySQL、DB2、PostgreSQL、SOQLite 等 各 种 主流 数据 库 提供 了 大 量 简明 的 实例 。 与 
其 他 同类 图 书 不 同 ， 它 没有 过 多 阐述 数据 库 基础 理论 ， 而 是 专门 针对 一 线 软件 开发 人 员 ， 直 接 从 SQL 
SELECT 开始 ， 讲 述 实 际 工 作 环境 中 最 常用 和 最 必需 的 SQL 知识 ， 实 用 性 极 强 。 通 过 本 书 ， 读 者 能 
从 没有 多 少 SQL 经 验 的 新 手 ， 迅 速 编写 出 世界 级 的 SQLI 
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方式 还 使 本 书 成 为 方便 的 快速 查询 手册 。 
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